我想根据文本的长度过滤我的模型
MyModel.objects.filter(len(text) > 10)
其中text是MyModel模型中的Char或Text字段
如果您只添加一个预先计算(记忆)文本长度的列,那将会更好更快。
EG
class MyModel(models.Model):
text = models.TextField()
text_len = models.PositiveIntegerField()
def save(self, *args, **kwargs):
self.text_len = len(self.text)
return super(MyModel, self).save(*args, **kwargs)
MyModel.objects.filter(text_len__gt = 10) # Here text_len is pre-calculated by us on `save`
对于Django> = 1.8,您可以使用Length function,这是@ Pratyush的CHAR_LENGTH()
用于MySQL,或LENGTH()
用于其他一些数据库:
from django.db.models.functions import Length
qs = MyModel.objects.annotate(text_len=Length('text_field_name')).filter(
text_len__gt=10)
另一种方式是:
MyModel.objects.extra(where=["CHAR_LENGTH(text) > 300"])
这可以在文本长度超过255个字符的情况下使用。
您可以使用正则表达式过滤器来搜索特定长度的文本:
MyModel.objects.filter(text__regex = r'.{10}.*')
警告:对于MySQL,最大长度值为255.否则抛出异常:
DatabaseError: (1139, "Got error 'invalid repetition count(s)' from regexp")
通过将内置函数Length
注册为CharField
查找的变换,可以为Django> = 1.9提供一个很好的解决方案。
在项目中注册一次转换。 (最好的地方可能是models.py。)
from django.db.models import CharField
from django.db.models.functions import Length
CharField.register_lookup(Length, 'length')
使用:
result = MyModel.objects.filter(text__length__gt=10)
在Length as a transform的文档中查看完全相同的示例。
它适用于所有后端,由LENGTH()
为大多数后端编译,CHAR_LENGTH()
为MySQL编译。然后自动为CharField的所有子类注册,例如对于EmailField。 TextField
必须单独注册。注册名称“length”是安全的,因为变换名称永远不会被同名的字段名称或相关字段名称着色或加阴影。
唯一的缺点可能是可读性难题:“长度”来自何处? (查找是全局的,但是如果在可读性方面有用,可以在更多模块中重复安全地重复注册,在查询运行时没有任何可能的开销。)
其他类似的有价值的解决方案是上面的hobs,如果注册计数并且如果不重复使用类似的查询则更短。
我会解决您的应用服务器上的问题,而不是对您的数据库征税。你可以这样做:
models_less_than_ten = []
mymodel = MyModel.objects.all()
for m in mymodel:
if len(m.text) > 10:
models_less_than_ten.append(m)