Как мне динамически получить класс модели Django?
-
03-07-2019 - |
Вопрос
Не имея полного пути к модулю модели Django, возможно ли сделать что-то вроде:
model = 'User' [in Django namespace]
model.objects.all()
... в отличие от:
User.objects.all().
Редактировать:Я пытаюсь выполнить этот вызов на основе ввода из командной строки.Можно ли избежать инструкции import, например,
model = django.authx.models.User
Без того, чтобы Django вернул ошибку:
"global name django is not defined."
Решение
Я думаю, что вы ищете это
from django.db.models.loading import get_model
model = get_model('app_name', 'model_name')
Конечно, есть и другие методы, но я бы справился с этим, если вы не знаете, какой файл моделей вам нужно импортировать в ваше пространство имен. (Обратите внимание, что на самом деле нет способа безопасно получить модель, не зная сначала, к какому приложению она принадлежит. Посмотрите исходный код файла load.py, если вы хотите проверить свою удачу при переборе всех моделей приложений.)
Обновление для Django 1.7 +: в соответствии с временная шкала устаревания , django.db.models.loading
устарела в Django 1.7 и будет удалена в Django 1.9. Как указано в ответе Alasdair ,
В Django 1.7+ есть реестр приложений . Вы можете использовать apps.get_model
метод для динамического получения модели:
from django.apps import apps
MyModel = apps.get_model('app_label', 'MyModel')
Другие советы
Для Django 1.7+ существует реестр приложений . Вы можете использовать apps.get_model
метод для динамического получения модели.
from django.apps import apps
MyModel = apps.get_model('app_label', 'MyModel')
from django.authx.models import User
model = User
model.objects.all()
модель = django.authx.models.Пользователь
?Django возвращает ошибку "глобальное имя django не определено".
Django не возвращает ошибку.Python так и делает.
Сначала вы ДОЛЖНЫ импортировать модель.Вы должны импортировать его с помощью
from django.authx.models import User
Во-вторых, если вы получите сообщение об ошибке, которое django
не определен, значит, Django установлен некорректно.У вас должен быть Django на вашем компьютере. PYTHONPATH
или установлен в вашей библиотеке Python / site-packages.
Чтобы правильно установить Django, см. http://docs.djangoproject.com/en/dev/intro/install/#intro-install
Классы являются "первым классом" объекты в Python, то есть они могут передаваться и управляться так же, как и все другие объекты.
Модели - это классы. Из того, что вы создаете новые модели, используя операторы классов, вы можете сказать:
class Person(models.Model):
last_name = models.CharField(max_length=64)
class AnthropomorphicBear(models.Model):
last_name = models.CharField(max_length=64)
Идентификаторы Person
и AnthropomorphicBear
связаны с классами Django, так что вы можете передавать их. Это может быть полезно, если вы хотите создать вспомогательные функции, которые работают на уровне модели (и имеют общий интерфейс):
def print_obj_by_last_name(model, last_name):
model_name = model.__name__
matches = model.objects.filter(last_name=last_name).all()
print('{0}: {1!r}'.format(model_name, matches))
Таким образом, print_obj_by_last_name
будет работать с моделями Person
или AnthropomorphicBear
. Просто передайте модель так:
print_obj_by_last_name(model=Person, last_name='Dole')
print_obj_by_last_name(model=AnthropomorphicBear, last_name='Fozzy')
Если имя модели передано в виде строки, думаю, одним из способов может быть
modelname = "User"
model = globals()[modelname]
Но в некоторых контекстах портирование с помощью globals () может быть немного опасным. Так что обращайтесь с осторожностью:)