1. 程式人生 > 程式設計 >重寫django的model下的objects模型管理器方式

重寫django的model下的objects模型管理器方式

django自定義管理器和model的繼承

在我寫程式碼時建立的很多資料庫需要一些共同的欄位,比如is_active,create_time這些欄位,所以可以建立一個基類model

模型的繼承

我們可以充分利用oop特性,建立一個基類model,以後所有的model都繼承這個類, 就不用在每個類都額外新增欄位

class BaseModel(models.Model):
  is_active = models.BooleanField(default=True)
  create_time = models.DateTimeField()

  class Meta:
   '''
   指定這個類是一個抽象模型類, 這個模型就不會被生成表
   '''
    abstract = True 
  
class ChildModel(BaseModel):
  name = models.CharField(max_length=10)
  age = models.PositiveIntegerField()
    
  class Meta:
    '''
    meta屬性可以重寫, 不重寫的話也會繼承基類meta
    '''
    verbose_name = '子模型'

重寫Django的model.objects

當我們使用model.objects.filter()時,經常會過濾掉is_active=False的,但是這樣我們每個 filter(is_active=True)都需要這樣寫,那我們能不能重寫掉這個filter方法,讓他每次可以自己過濾掉,這裡我們可以看一下django的原始碼

# 這裡原始碼我就不拉出來了, 我們可以看到django是set了個objects屬性, 這個屬性對應是一個類的例項物件

class Manager(BaseManager.from_queryset(QuerySet)):
'''
objects 是這個類的例項物件,這個類我們可以看到他其實繼承了 QuerySet類裡的方法,所以我們可以直接重寫個這個類方法,並重寫filter方法
'''
  def filter(self,*args,**kwargs):
    """
    我們可以看到 QuerySet裡的filter方法和exclude方法都是呼叫了這個方法,這個方法的內部實現可
    以自己看一看,就是用的kwargs裡的引數
    """
    return self._filter_or_exclude(False,**kwargs)

class MyManager(models.Manager):
  
  def filter(self,**kwargs):
  '''
  原始碼的filter函式是有返回值的,所以我們將super出來的結果返回出去,我們只需要在kwargs中增加個引數就行
  '''
    if not kwargs.get('is_active',True): # 如果需要檢視所有資料,
      kwargs['is_active'] = False
    return super(AddressManager,self).filter(*args,**kwargs)

# 最後我們在我們的ChildModel下將objects物件賦值為MyManager的例項物件就可以使用了
# 我們可以將這個objects寫在基類裡面,這樣所有的model就可以都會有這個方法,具體怎麼使用視情況而定

class ChildModel(models.Model):
  objects = MyManager()

模型類也是可以多繼承的,如果有特殊的用法可以具體百度或Google一下, 這裡只是個簡單的引入面向物件的概念,我們很多地方可以用繼承框架已有的類並重寫類中的方法,是我們的程式碼更加pythonic

補充知識:Django中的Model中的欄位屬性和選項

欄位型別:

autoField:一個根據實際ID自動增長的IntegerField,通常不指定,如果不指定,一個主鍵欄位將自動新增到模型中

CharField(max_length=字元長度):字串,預設的表彰樣式是TextInput

TextField:大文字欄位,一般超過4000時使用,預設的表單控制元件是Textarea

IntegerField:整數

DecimalField(max_digits=None,decimal_places=None): 使用Python的Decimal例項表示的十進位制浮點數

引數說明: DecimalField.max_digits----位數總數
DecimalField.decimal_places---小數點後的數字位置

FloatField: 使用Python的float例項來表示的浮點數

BooleanField:True/False 欄位,此欄位的預設表彰控制是CheckboxInput

NullBooleanField:支援 Null,True,False 三種值

DateField([auto_now=False,auto_now_add=False]):使用Python的datetime.date例項表示的日期

引數說明:
DateField.auto_now
每次儲存物件時,自動設定該欄位為當前時間,用於“最後一次修改”
的時間戳,它總是使用當前日期,預設為 False

DateField.auto_now_add

當前物件第一次被建立時自動設定當前時間,用於建立的時間戳,

它總是使用當前日期,預設為 False

說明
該欄位預設對應的表單控制元件是一個TextInput.在管理員站點添加了一個
JavaScript寫的日曆控制元件,和一個“Today”的快捷按鈕,包含了一個額外
的invalid_date錯誤訊息鍵
注意
auto_now_add,auto_now,and default 這些設定是相互排斥的,他們之間
的任何組合將會發生錯誤的結果

TimeField: 使用Python的datetime.time例項表示的時間,引數同DateField

DateTimeField: 使用Python的datetime, datetime例項表示的日期和時間,引數同DateField

FileField: 一個上傳檔案的欄位

ImageField:繼承了FileField的所有屬性和方法,但對上傳的物件進行校驗,確保它是一個有效的image

欄位選項

概述:通過欄位選項,可以實現對欄位的約束, 在欄位物件中通過關鍵字引數指定

null:如果為True,Django將空值以NULL儲存在資料庫中,預設值為False

blanke:如果為True,則該欄位允許為空白,預設值為False

注意: null是資料庫範疇的概念,blank是表彰驗證範疇的概念

db_column:欄位的名稱,如果未指定,則使用屬性的名稱

db_index:若值為 True,則在表中會為此欄位建立索引

default: 預設值

primary_key: 若為 True,則該欄位會成為模型的主鍵欄位

unique:如果為 True,這個欄位在表中必須有唯一值

注意:在生成遷移檔案之後如果修改的引數不影響表結構,則不用重新生成遷移檔案。(default,blank 不影響表的結構。不用重新生成遷移檔案)

關係型別

分類:

ForeignKey:一對多,將欄位定義在多的端中

ManyToManyField:多對多,將欄位定義在兩端中

OneToOneField:一對一,將欄位定義在任意一端中

(1)用一訪問多:格式: 物件.模型類小寫_set

示例: grade.students_set

(2)用一訪問一:格式:物件.模型類小寫

示例: grade.studnets

(3)訪問id:格式:物件.屬性_id

例子:btitle = models.CharField(max_length=20,db_column=‘title',unique=‘true')

以上這篇重寫django的model下的objects模型管理器方式就是小編分享給大家的全部內容了,希望能給大家一個參考,也希望大家多多支援我們。