我想通过一个常见的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()进行猴子修补会起作用吗,或者它是反样式,因此不应使用?
可以使用动态字段名称创建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)
尝试使用工厂模式来设置不同版本的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)
归档时间: |
|
查看次数: |
542 次 |
最近记录: |