1. 程式人生 > >Manager isn't accessible via %s instances" % cls.__name_ 報錯資訊

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 四個重要類:
ModelQuerySetQueryObjects
QuerySet 主要是定義了一些介面,如 filter, count 等。它是惰性求值的,它並不直接求出值,只是在需要的時候查詢。
Query 實現 sql 的拼接,它將語句交給 sql 編譯物件。

一次查詢過程:


Django 模型查詢過程
如果還想深入,可以看看  QuerySet 的原始碼。
ps: objects 是一個  Manager 例項,而
Manager 是一個空殼,它繼承自  QuerySet