我已经定义了一个User类,它(最终)继承自models.Model。我想获得为这个模型定义的所有字段的列表。例如,phone_number = CharField(max_length=20)。基本上,我想检索从Field类继承的任何东西。

我认为我可以利用inspect.getmembers(model)来检索这些字段,但是它返回的列表不包含这些字段中的任何一个。看起来Django已经掌握了这个类,添加了它所有神奇的属性,并去掉了实际定义的内容。所以…我怎样才能得到这些田地?他们可能有一个功能来检索他们自己的内部目的?


当前回答

MyModel._meta.get_all_field_names()在之前的几个版本中被弃用,并在Django 1.10中被移除。

以下是文档中向后兼容的建议:

from itertools import chain

list(set(chain.from_iterable(
    (field.name, field.attname) if hasattr(field, 'attname') else (field.name,)
    for field in MyModel._meta.get_fields()
    # For complete backwards compatibility, you may want to exclude
    # GenericForeignKey from the results.
    if not (field.many_to_one and field.related_model is None)
)))

其他回答

至少在Django 1.9.9(我目前使用的版本)中,请注意.get_fields()实际上也将任何外部模型“视为”字段,这可能是有问题的。假设你有:

class Parent(models.Model):
    id = UUIDField(primary_key=True)

class Child(models.Model):
    parent = models.ForeignKey(Parent)

由此可见

>>> map(lambda field:field.name, Parent._model._meta.get_fields())
['id', 'child']

,如@Rockallite所示

>>> map(lambda field:field.name, Parent._model._meta.local_fields)
['id']

MyModel._meta.get_all_field_names()在之前的几个版本中被弃用,并在Django 1.10中被移除。

以下是文档中向后兼容的建议:

from itertools import chain

list(set(chain.from_iterable(
    (field.name, field.attname) if hasattr(field, 'attname') else (field.name,)
    for field in MyModel._meta.get_fields()
    # For complete backwards compatibility, you may want to exclude
    # GenericForeignKey from the results.
    if not (field.many_to_one and field.related_model is None)
)))

这里提到的get_all_related_fields()方法在1.8中已弃用。从现在开始,它是get_fields()。

>> from django.contrib.auth.models import User
>> User._meta.get_fields()

结合给定线程的多个答案(谢谢!),并提出以下通用解决方案:

class ReadOnlyBaseModelAdmin(ModelAdmin):
    def has_add_permission(self, request):
        return request.user.is_superuser

    def has_delete_permission(self, request, obj=None):
        return request.user.is_superuser

    def get_readonly_fields(self, request, obj=None):
        return [f.name for f in self.model._meta.get_fields()]

所以在我找到这篇文章之前,我成功地找到了这个方法。

Model._meta.fields

它的工作原理和

Model._meta.get_fields()

我不确定结果有什么不同,如果有的话。我运行了这个循环,得到了相同的输出。

for field in Model._meta.fields:
    print(field.name)