4

当您使用defer(...)查询命令时,Django 返回的类与您的原始模型不同。使用 defer 字段时如何动态获取模型的名称?

在代码中:

obj_nodefer = model_class.objects.filter(title="foo")[0]
model_name = str(type(obj_nodefer)) # Works just fine

obj_defer = model_class.objects.filter(title="foo").defer("content")[0]
model_name = str(type(obj_defer)) # Does't return the right name because of defer above.

如何从中获取模型的名称obj_defer

4

4 回答 4

6

在延迟类上,您还可以在不__class__.__name__使用字符串的情况下获取原始类:

obj._meta.concrete_model
'product.models.ProductModelName'

obj._meta.concrete_model._meta.app_label
'product'

obj._meta.concrete_model._meta.model_name
'productmodelname'
于 2015-11-24T15:19:44.720 回答
4

要获得非延迟模型,请执行

t = type(obj)
if t._deferred:
  t = t.__base__
于 2014-01-09T23:00:29.923 回答
0

您可以访问,obj.__class__

  >>> print obj.__class__
  <class 'product.models.Product_Deferred_name'>
  >>> print obj.__class__.__name__
  Product_Deferred_name
  >>> print obj.__class__.__name__.split("_")[0]
  Product

如果您的型号名称包含_,

  >>> print obj.__class__.__name__.replace("_Deferred_name","")
  Product
于 2013-01-12T19:31:16.970 回答
0

在 django 1.4 中,延迟类的构造如下:

def deferred_class_factory(model, attrs):
    """
    Returns a class object that is a copy of "model" with the specified "attrs"
    being replaced with DeferredAttribute objects. The "pk_value" ties the
    deferred attributes to a particular instance of the model.
    """
    class Meta:
        proxy = True
        app_label = model._meta.app_label

    # The app_cache wants a unique name for each model, otherwise the new class
    # won't be created (we get an old one back). Therefore, we generate the
    # name using the passed in attrs. It's OK to reuse an existing class
    # object if the attrs are identical.
    name = "%s_Deferred_%s" % (model.__name__, '_'.join(sorted(list(attrs))))
    name = util.truncate_name(name, 80, 32)

    overrides = dict([(attr, DeferredAttribute(attr, model))
            for attr in attrs])
    overrides["Meta"] = Meta
    overrides["__module__"] = model.__module__
    overrides["_deferred"] = True
    return type(name, (model,), overrides)

所以,理论上你应该通过做

type(type(deferred))
于 2013-01-12T20:13:36.897 回答