django自定义Field实现一个字段存储以逗号分隔的字符串
要实现一个以逗号分隔的字符串字段,可以使用Django的自定义Field来实现。
步骤如下:
1. 创建一个新的Django App
首先要创建一个新的Django应用程序,例如 "comma_field"。
使用以下命令创建:
$ python manage.py startapp comma_field
2. 定义一个CommaSeparatedField类
在新创建的应用程序的 models.py
文件中,定义一个 CommaSeparatedField
类,继承自 Django 的 Field
类,然后实现 .db_type()
方法和 .to_python()
和 .get_prep_value()
方法。其中:
.db_type()
方法返回字段所使用的数据库类型,这里设为VARCHAR(255)
;.to_python()
将从数据库中获取的数据转换为Python对象;.get_prep_value()
将Python对象转换为Django可用的值。
from django.db import models
class CommaSeparatedField(models.Field):
def db_type(self, connection):
return 'VARCHAR(255)'
def to_python(self, value):
if value is None:
return []
return [item.strip().lower() for item in value.split(',')]
def get_prep_value(self, value):
if value is None:
return None
return ','.join([str(s) for s in value])
3. 在模型中使用
在需要使用以逗号分隔的字符串字段的模型中,导入刚定义好的 CommaSeparatedField
类,然后在定义字段时直接使用 CommaSeparatedField
类。
from django.db import models
from comma_field.models import CommaSeparatedField
class MyModel(models.Model):
my_field = CommaSeparatedField()
# 其他字段省略
示例1. 字符串的写入和读取:
obj = MyModel.objects.create(my_field='A, b, c, d') # 写入
assert obj.my_field == ['a', 'b', 'c', 'd'] # 读取
示例2. 使用in
查询操作符:
如果需要使用 in
查询操作符的话,需要自定义过滤条件表达式,将整个字符串用逗号分割后,与某个值逐个比较。
from django.db.models.lookups import Exact
class CommaSeparatedExact(Exact):
def process_rhs(self, compiler, connection):
rhs, rhs_params = super().process_rhs(compiler, connection)
return "({})".format(','.join(["%s"] * len(rhs_params))), rhs_params
MyModel.objects.filter(my_field__in=['a', 'c'], my_field__exact=CommaSeparatedExact('b'))
# SELECT * FROM myapp_mymodel WHERE my_field IN ('a', 'c') AND my_field = 'b'