具有动态名称的Django模型字段

MrB*_*Win 4 python django

我想通过一个常见的mixin或抽象模型向现有模型添加新的CharField,但是这些字段的名称取决于配置。因此,一个模型将具有someprefix1_title字段,而另一模型将具有someprefix2_title字段。

是否可以使这种方法起作用:

class AbstractModel(models.Model):
    self.fields_prefix + '_title' = models.CharField(max_length=255, blank=True, default='')

    class Meta:
        abstract = True

class ModelOne(AbstractModel):
    fields_prefix = 'someprefix1'
    id = models.AutoField(primary_key=True)

class ModelTwo(AbstractModel):
    fields_prefix = 'someprefix2'
    id = models.AutoField(primary_key=True)
Run Code Online (Sandbox Code Playgroud)

因此ModelOne可以具有ID和someprefix1_title字段。

upd:用add_to_class()进行猴子修补会起作用吗,或者它是反样式,因此不应使用?

Moh*_*han 5

可以使用动态字段名称创建Django模型。这是一个简单的Django模型:

class Animal(models.Model):
    name = models.CharField(max_length=32)
Run Code Online (Sandbox Code Playgroud)

这是使用构造的等效类type()

attrs = {
    'name': models.CharField(max_length=32),
    '__module__': 'myapp.models'
}
Animal = type("Animal", (models.Model,), attrs)
Run Code Online (Sandbox Code Playgroud)

可以使用常规方式定义的任何Django模型均可使用type()

要运行迁移,South提供了一组可靠的函数来处理Django项目的模式和数据库迁移。在开发中使用时,South可以建议迁移,但不会尝试自动应用它们

from south.db import db
model_class = generate_my_model_class()
fields = [(f.name, f) for f in model_class._meta.local_fields]
table_name = model_class._meta.db_table
db.create_table(table_name, fields)
# some fields (eg GeoDjango) require additional SQL to be executed
db.execute_deferred_sql()
Run Code Online (Sandbox Code Playgroud)

  • @SimonCharette抱歉,这不是我最好的一天。你当然是对的。 (2认同)

whp*_*whp 5

尝试使用工厂模式来设置不同版本的AbstractModel.

通过这种方式,可以更严格地控​​制AbstractModel通过工厂函数方式修改的方式dynamic_fieldname_model_factory

我们也不会修改ModelOne或修改ModelTwo它们的定义——其他解决方案指出这有助于避免可维护性问题。

模型.py:

from django.db import models


def dynamic_fieldname_model_factory(fields_prefix):
    class AbstractModel(models.Model):

        class Meta:
            abstract = True

    AbstractModel.add_to_class(
        fields_prefix + '_title',
        models.CharField(max_length=255, blank=True, default=''),
    )
    return AbstractModel


class ModelOne(dynamic_fieldname_model_factory('someprefix1')):
    id = models.AutoField(primary_key=True)


class ModelTwo(dynamic_fieldname_model_factory('someprefix2')):
    id = models.AutoField(primary_key=True)
Run Code Online (Sandbox Code Playgroud)

这是此代码生成的迁移:

# Generated by Django 2.1.7 on 2019-03-07 19:53

from django.db import migrations, models


class Migration(migrations.Migration):

    initial = True

    dependencies = [
    ]

    operations = [
        migrations.CreateModel(
            name='ModelOne',
            fields=[
                ('someprefix1_title', models.CharField(blank=True, default='', max_length=255)),
                ('id', models.AutoField(primary_key=True, serialize=False)),
            ],
            options={
                'abstract': False,
            },
        ),
        migrations.CreateModel(
            name='ModelTwo',
            fields=[
                ('someprefix2_title', models.CharField(blank=True, default='', max_length=255)),
                ('id', models.AutoField(primary_key=True, serialize=False)),
            ],
            options={
                'abstract': False,
            },
        ),
    ]
Run Code Online (Sandbox Code Playgroud)