您好,登錄后才能下訂單哦!
ORM 查詢管理器
對于 ORM 定義: 對象關系映射, Object Relational Mapping, ORM, 是一種程序設計技術,用于實現面向對象編程語言里不同類型系統的數據之間的轉換。從效果上說,它其實是創建了一個可在編程語言里使用的“虛擬對象數據庫”。ORM 能大大簡化并抽象數據庫的操作.
假設 django 的一個工程中包含一個名為 Book 的模塊(model), 在 views.py 的函數中可能會寫出查詢語句:
# views.py def index(request): book_set = Book.objects.filter(id=1) 或者 book_set = Book.objects.all() ......
ORM 的作用就是這樣, 并不需要寫更復雜的 SQL 語句, 所有的的事情都被 ORM 代勞了.
上面中, Book 實際上是一個 Model 實例, 但先是從 Book.objects 開始說起. Book.objects 實際上是一個 Manager 類實例, 每個 Model 都會有一個, 用戶的查詢操作幾乎是從這里開始的. 萬萬可以將 Model 實例理解為關系表中的一個表項數據, 而 Manager 實例可以理解數據庫查詢的入口.
實際上, 無論如何都在 Model 類的源碼中找到任何 objects 屬性的字眼, 因此它肯定是在某個時間點上增加的. 可以在 django.db.models.manager.py 中找到下面的函數:
這個函數確保每一個 model 都有一個管理器 objects
def ensure_default_manager(sender, **kwargs): ...... if not getattr(cls, '_default_manager', None): # Create the default manager, if needed. try: cls._meta.get_field('objects') raise ValueError("Model %s must specify a custom Manager, because it has a field named 'objects'" % cls.__name__) except FieldDoesNotExist: pass """ 關鍵的一步, 將一個 Manager 實例掛鉤到 cls.objects, 將 model.add_to_class() 方法羅列如下; def add_to_class(cls, name, value): if hasattr(value, 'contribute_to_class'): value.contribute_to_class(cls, name) else: setattr(cls, name, value) 關鍵是 Manager 有 contribute_to_class() 方法, 由此看來, model.objects 并不是一個 Manager 實例, 實際上他是一個 ManagerDescriptor 實例. """ cls.add_to_class('objects', Manager()) cls._base_manager = cls.objects elif not getattr(cls, '_base_manager', None): default_mgr = cls._default_manager.__class__ if (default_mgr is Manager or getattr(default_mgr, "use_for_related_fields", False)): cls._base_manager = cls._default_manager else: # Default manager isn't a plain Manager class, or a suitable # replacement, so we walk up the base class hierarchy until we hit # something appropriate. for base_class in default_mgr.mro()[1:]: if (base_class is Manager or getattr(base_class, "use_for_related_fields", False)): cls.add_to_class('_base_manager', base_class()) return
由此可以發現, Model.objects 在這個時候被添加了. 因此用戶可以在代碼中使用 Book.objects. 至于這個函數在何時被調用, 待后面會詳述 django 內部的信號機制. 暫且你可以將其理解為在 django 服務器啟動的時候, 這些會被自動調用就好了.
Manager 實現
Manager 保護技法
如果可以在 book_set = Book.objects.filter(id=1) 這一句上設置斷點, 并 step into 的時候, 發現 Book.objects 實際上的實際上不是一個 Manager 實例, 而是一個 ManagerDescriptor 實例, 這是 django 特意為 Manager 做的一層包裝. 為什么要這么做 ?
django 規定, 只有 Model 類可以使用 objects, Model 類實例不可以. 請注意區分類和類實例之間的區別.
我認為這樣做是有道理的. Book.objects.filter(id=1) 返回的是 QuerySet 對象, 而 QuerySet 對象可以看成是 Model 實例的集合, 也就是 book_set 是 Model 實例的集合. 假使 「Model 類的實例可以使用 objects 屬性」, 即「從一本書中查詢書」這在語意上不通過. 只能是「從書的集合(Book)中查詢書」.
所以 django 用 ManagerDescriptor 特意為 Manager 做的一層包裝. 可以在 django.db.models.manager.py 中找到
ManagerDescriptor 的實現:
class ManagerDescriptor(object): """
很經典的掩飾, 為 Manager 特殊設定 Descriptor, 從而, 只能讓類訪問, 而不能讓類的實例來訪問. 具體是靠 __get__(self, instance, type=None) 方法來實現來的: 第二個參數 instance, 當 class.attr 的時候, instance 為 None; 當 obj.attr 的時候, instance 為 obj.
""" # This class ensures managers aren't accessible via model instances. # For example, Poll.objects works, but poll_obj.objects raises AttributeError. def __init__(self, manager): self.manager = manager def __get__(self, instance, type=None): if instance != None: raise AttributeError("Manager isn't accessible via %s instances" % type.__name__) return self.manager
所要詳述的是 __get__() 函數. python 的語法里有修飾器(descriptor)這么一說, 而 python 的屬性類型就是這么實現的. descriptor 實現 __get__(), __set__(), 接著將其添加到一個類中. 譬如下面的例子:
class Celsius(object): def __init__(self, value=0.0): self.value = float(value) def __get__(self, instance, owner): print instance,owner return self.value def __set__(self, instance, value): print instance,value self.value = float(value) class Temperature(object): celsius = Celsius() t = Temperature() t.celsius Temperature.celsius
當對 descriptor 賦值的時候, 其本身 __set__ 會被調用, 取值的時候 __get__() 會被調用. __set__,__get__ 函數的 instance 參數即為類實例(所以, t.cellsius 調用 __get__() 的時候, instance 參數是 t, Temperature.celsius 調用 __get__() 的時候, instance 參數是 Temperature).
所以, 可以通過判斷 instance 來判斷調用者是否是類實例. 也就由此可以拒絕類實例的訪問, 發現 ManagerDescriptor 就是這么實現的.
總結
Book.objects 實際上是一個 Manager, 實際上的實際上卻是一個 ManagerDescriptor, 但真正起作用的還是 Manager, ManagerDescriptor 是修飾器, 是 django 的保護技法.
從 Manager 的實現來看, 它的多數函數會返回 QuerySet 對象, 而且透漏了一個重點: QuerySet 對象可以看成是 Model 實例的集合.
我已經在 github 備份了 Django 源碼的注釋: Decode-Django, 有興趣的童鞋 fork 吧.
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持億速云。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。