Manager isn't accessible via %s instances" % cls.__name_ 報錯資訊
Django 的 orm 中使用到了元類和描述符這些高階知識,瞭解一下的可以看看這篇文章。
元類其實就是用來定義類的,我的理解是這樣的:當很多類有相同的屬性,那麼就可以提取這些相同的屬性到一個類中,元類就是用來封裝那些的,或者給某些類新增一些屬性,定製類。你看看 Django model
寫的程式碼有多少,而背後元類默默做了很多東西。
Django model
需要繼承自 models.Model
,跟蹤進去發現 class Model(metaclass=ModelBase):
,Model
其實是根據 ModelBase
構建的,在例項化的時候,會首先執行 ModelBase
的 __new__
_meta
這麼一個屬性,具體的可以看看原始碼。
我們查詢資料的時候一直用到 obejcts
方法,那麼它是哪來的呢?在封裝self._meta
後呼叫了 _prepare
,在它裡面發現了一個管理器 manager
。
if not opts.managers: if any(f.name == 'objects' for f in opts.fields): raise ValueError( "Model %s must specify a custom Manager, because it has a" "field named 'objects'." % cls.__name__ ) manager = Manager() manager.auto_created = True cls.add_to_class('objects', manager)
這裡使用了 manager
,然後註冊了 objects
。那麼來看看 Manager
:
class Manager(BaseManager.from_queryset(QuerySet)): pass
什麼也沒做,只是根據 BaseManager
QuerySet
,到這裡,你就大體明白了吧,我們一般查詢出來的都是一個 QuerySet
物件。
@classmethod def from_queryset(cls, queryset_class, class_name=None): if class_name is None: class_name = '%sFrom%s' % (cls.__name__, queryset_class.__name__) class_dict = { '_queryset_class': queryset_class, } class_dict.update(cls._get_queryset_methods(queryset_class)) return type(class_name, (cls,), class_dict)
這裡使用了 type
來創造一個類。
其實準確來說,Manager
應該是繼承了兩個類 BaseManager
和 QuertSet
。
再來看看 add_to_class
:
def add_to_class(cls, name, value): # We should call the contribute_to_class method only if it's bound if not inspect.isclass(value) and hasattr(value, 'contribute_to_class'): value.contribute_to_class(cls, name) else: setattr(cls, name, value)
這裡又呼叫了 manager
的 contribute_to_class
方法:
def contribute_to_class(self, model, name): if not self.name: self.name = name self.model = model setattr(model, name, ManagerDescriptor(self)) model._meta.add_manager(self)
又給 objects
賦值了一個 ManagerDescriptor
例項,這個是幹嘛的呢?屬性描述符:
class ManagerDescriptor: def __init__(self, manager): self.manager = manager def __get__(self, instance, cls=None): if instance is not None: raise AttributeError("Manager isn't accessible via %s instances" % cls.__name__) if cls._meta.abstract: raise AttributeError("Manager isn't available; %s is abstract" % ( cls._meta.object_name, )) if cls._meta.swapped: raise AttributeError( "Manager isn't available; '%s.%s' has been swapped for '%s'" % ( cls._meta.app_label, cls._meta.object_name, cls._meta.swapped, ) ) return cls._meta.managers_map[self.manager.name]
這裡做了一些限制。
Django orm
四個重要類:
Model
,QuerySet
,Query
,Objects
。
QuerySet
主要是定義了一些介面,如 filter
, count
等。它是惰性求值的,它並不直接求出值,只是在需要的時候查詢。
Query
實現 sql
的拼接,它將語句交給 sql
編譯物件。
一次查詢過程:
Django 模型查詢過程
如果還想深入,可以看看
QuerySet
的原始碼。
ps:
objects
是一個
Manager
例項,而
Manager
是一個空殼,它繼承自
QuerySet
。