Django:从字符串中获取模型?

Posted

技术标签:

【中文标题】Django:从字符串中获取模型?【英文标题】:Django: Get model from string? 【发布时间】:2011-06-20 09:03:24 【问题描述】:

在 Django 中,您可以指定如下关系:

author = ForeignKey('Person')

然后它必须在内部将字符串“Person”转换为模型Person

执行此操作的函数在哪里?我想用它,但我找不到它。

【问题讨论】:

【参考方案1】:

从 Django 1.11 到 4.0(至少),它是 AppConfig.get_model(model_name, require_ready=True)

从 Django 1.9 开始,该方法为 django.apps.AppConfig.get_model(model_name)。 -- danihp

从 Django 1.7 开始,django.db.models.loading 已弃用(将在 1.9 中删除)以支持新的应用程序加载系统。 -- Scott Woodall


找到了。它在这里定义:

from django.db.models.loading import get_model

定义为:

def get_model(self, app_label, model_name, seed_cache=True):

【讨论】:

此解决方案在 Django 1.7 中已弃用,请参阅此解决方案的其他答案:***.com/a/26126935/155987 嗨@mpen,我建议你更新你的答案。 On django 1.9 get_model is defined on AppConfig: from django.apps import AppConfig 在 Django 1.7+ 中,最好在 Django 应用程序注册表上使用 get_model(),它可以通过 django.apps.apps.get_model(model_name) 获得。 AppConfig 对象有不同的用途,需要您创建一个 AppConfig 实例才能调用get_model() Django 1.11 需要@luke_aus 提供的答案 最新 (2020) 解决方案:from django.apps import apps 然后:apps.get_model('app_name', 'Model') ***.com/questions/38443628/…【参考方案2】:

django.db.models.loading 是 deprecated in Django 1.7 (removed in 1.9) 支持新的 application loading system。

Django 1.7 docs 改为给我们以下内容:

>>> from django.apps import apps
>>> User = apps.get_model(app_label='auth', model_name='User')
>>> print(User)
<class 'django.contrib.auth.models.User'>

【讨论】:

我曾经使用 pydoc 的“定位”功能...不确定这里的区别,但为了留在 Django 中,我已经切换到这种方法。谢谢。【参考方案3】:

仅适用于任何被卡住的人(就像我一样):

from django.apps import apps

model = apps.get_model('app_name', 'model_name')

app_name 应使用引号列出,model_name 也应使用引号(即不要尝试导入它)

get_model 接受小写或大写 'model_name'

【讨论】:

【参考方案4】:

大多数模型“字符串”以“appname.modelname”的形式出现,因此您可能希望在 get_model 上使用此变体

from django.db.models.loading import get_model

your_model = get_model ( *your_string.split('.',1) )

django 代码中通常将此类字符串转换为模型的部分稍微复杂一些来自django/db/models/fields/related.py

    try:
        app_label, model_name = relation.split(".")
    except ValueError:
        # If we can't split, assume a model in current app
        app_label = cls._meta.app_label
        model_name = relation
    except AttributeError:
        # If it doesn't have a split it's actually a model class
        app_label = relation._meta.app_label
        model_name = relation._meta.object_name

# Try to look up the related model, and if it's already loaded resolve the
# string right away. If get_model returns None, it means that the related
# model isn't loaded yet, so we need to pend the relation until the class
# is prepared.
model = get_model(app_label, model_name,
                  seed_cache=False, only_installed=False)

对我来说,这似乎是一个很好的案例,可以将其拆分为核心代码中的单个函数。但是,如果您知道您的字符串是“App.Model”格式,那么上面的两个衬里就可以了。

【讨论】:

我认为第二行应该是:your_model = get_model(*your_string.rsplit('.', 1))。应用标签有时是点格式,但型号名称始终是有效标识符。 看起来这在新的apps.get_model 中是没有必要的。 “作为一种快捷方式,此方法还接受 app_label.model_name 形式的单个参数。”【参考方案5】:

在 Django 1.7+ 中这样做的幸运方式是:

import django
model_cls = django.apps.apps.get_model('app_name', 'model_name')

因此,在所有框架教程的规范示例中:

import django
entry_cls = django.apps.apps.get_model('blog', 'entry')  # Case insensitive

【讨论】:

【参考方案6】:

2020 年解决方案:

from django.apps import apps

apps.get_model('app_name', 'Model')

根据你的例子:

apps.get_model('people', 'Person')

每: Import Error :cannot import name get_model

【讨论】:

【参考方案7】:

如果您不知道您的模型存在于哪个应用中,您可以这样搜索:

from django.contrib.contenttypes.models import ContentType 
ct = ContentType.objects.get(model='your_model_name') 
model = ct.model_class()

记住 your_model_name 必须是小写的。

【讨论】:

【参考方案8】:

另一种代码更少的版本,适合懒人。在 Django 2+ 中测试

from django.apps import apps
model = apps.get_model("appname.ModelName") # e.g "accounts.User"

【讨论】:

【参考方案9】:

我不确定它在 Django 中是在哪里完成的,但你可以这样做。

通过反射将类名映射到字符串。

classes = [Person,Child,Parent]
def find_class(name):
 for clls in classes:
  if clls.__class__.__name__ == name:
   return clls

【讨论】:

是 clls.__class__.__name__ 还是 clls.__name__?【参考方案10】:

这是一种从字符串中获取类的不太特定于 django 的方法:

mymodels = ['ModelA', 'ModelB']
model_list = __import__('<appname>.models', fromlist=mymodels)
model_a = getattr(model_list, 'ModelA')

或者您可以使用 importlib,如图所示 here:

import importlib
myapp_models = importlib.import_module('<appname>.models')
model_a = getattr(myapp_models, 'ModelA')

【讨论】:

以上是关于Django:从字符串中获取模型?的主要内容,如果未能解决你的问题,请参考以下文章

ORA-01861: 在 django 中执行获取模型对象时文字与格式字符串不匹配

django - 模型 unicode() 显示外键对象属性

获取 django 模型的类名

在 Django Rest Framework 中将整数的 JSON 列表转换为字符串

如何从Django的这个字典字符串中获取值?

如何从名称空间的 Django URL 中获取绝对字符串?