我有 first_name
、last_name
和alias
(可选)我需要搜索。所以,我需要一个查询来给我所有具有别名集的名称。
只要我能做到:
Name.objects.filter(alias!="")
那么,与上面的等价物是什么?
你可以这样做:
Name.objects.exclude(alias__isnull=True)
如果您需要排除空值和空字符串,最好的方法是将条件链接在一起,如下所示:
Name.objects.exclude(alias__isnull=True).exclude(alias__exact='')
将这些方法链接在一起基本上独立检查每个条件:在上面的示例中,我们排除了 alias
为空 或 为空字符串的行,因此您将获得所有具有 not- 的 Name
对象空,非空 alias
字段。生成的 SQL 如下所示:
SELECT * FROM Name WHERE alias IS NOT NULL AND alias != ""
您还可以将多个参数传递给对 exclude
的单个调用,这将确保仅排除满足 every 条件的对象:
Name.objects.exclude(some_field=True, other_field=True)
在这里,some_field
和 other_field
为真的行被排除,因此我们得到两个字段都不为真的所有行。生成的 SQL 代码看起来有点像这样:
SELECT * FROM Name WHERE NOT (some_field = TRUE AND other_field = TRUE)
或者,如果您的逻辑比这更复杂,您可以使用 Django 的 Q objects:
from django.db.models import Q
Name.objects.exclude(Q(alias__isnull=True) | Q(alias__exact=''))
有关详细信息,请参阅 Django 文档中的 this page 和 this page。
顺便说一句:我的 SQL 示例只是一个类比——实际生成的 SQL 代码可能看起来不同。通过实际查看它们生成的 SQL,您将更深入地了解 Django 查询的工作原理。
Name.objects.filter(alias__gt='',alias__isnull=False)
alias__isnull=False
条件是多余的。如果该字段是 Null
肯定会被第一个子句排除?
alias__gt
是唯一适用于我想从 JSON 中排除空字符串(如 {'something:''}
)的 JSON 类型列。所以工作语法是:jsoncolumnname__something__gt=''
首先,Django 文档强烈建议不要对基于字符串的字段(例如 CharField 或 TextField)使用 NULL 值。阅读文档以获取说明:
https://docs.djangoproject.com/en/dev/ref/models/fields/#null
解决方案:我认为,您也可以将 QuerySets 上的方法链接在一起。尝试这个:
Name.objects.exclude(alias__isnull=True).exclude(alias="")
这应该给你你正在寻找的集合。
1. 使用 exclude 时,请记住以下几点以避免常见错误:
应该不将 多个 条件添加到 一个 exclude()
块中,如 filter()
。要排除多个条件,您应该使用 multiple exclude()
。
示例:(NOT a AND NOT b)
Entry.objects.exclude(title='').exclude(headline='')
等于
SELECT... WHERE NOT title = '' AND NOT headline = ''
==================================================== ====
2.只有在你真正了解它的时候才使用multiple:
示例:NOT (a AND b)
Entry.objects.exclude(title='', headline='')
等于
SELECT.. WHERE NOT (title = '' AND headline = '')
从 Django 1.8 开始,
from django.db.models.functions import Length
Name.objects.annotate(alias_length=Length('alias')).filter(alias_length__gt=0)
如果要排除 null (None
)、空字符串 (""
) 以及包含空格的字符串 (" "
),您可以使用 __regex
和 __isnull
过滤器选项
Name.objects.filter(
alias__isnull = False,
alias__regex = r"\S+"
)
alias__isnull=False
排除所有列 null 列
aliax__regex = r"\S+"
确保列值至少包含一个或多个非空白字符。
你可以简单地这样做:
Name.objects.exclude(alias="").exclude(alias=None)
真的就是这么简单。 filter
用于匹配,而 exclude
用于匹配除指定内容之外的所有内容。这将评估为 SQL 为 NOT alias='' AND alias IS NOT NULL
。
alias=""
) 和 NULL (alias=None
) 别名。您的将包括具有 Name(alias=None)
的实例。
.filter(alias!="")
而不是标题的内容。我已经编辑了我的答案。但是,字符字段不应允许 NULL 值并将空字符串用于非值(按照惯例)。
这是另一种简单的方法。
Name.objects.exclude(alias=None)
None
与 ""
不同。
另一种使用通用 isempty
查找的方法,可用于任何字段。
django rest_framework 或其他使用 django 查找的应用程序也可以使用它:
from distutils.util import strtobool
from django.db.models import Field
from django.db.models.lookups import BuiltinLookup
@Field.register_lookup
class IsEmpty(BuiltinLookup):
lookup_name = 'isempty'
prepare_rhs = False
def as_sql(self, compiler, connection):
sql, params = compiler.compile(self.lhs)
condition = self.rhs if isinstance(self.rhs, bool) else bool(strtobool(self.rhs))
if condition:
return "%s IS NULL or %s = ''" % (sql, sql), params
else:
return "%s <> ''" % sql, params
然后你可以像这样使用它:
Name.objects.filter(alias__isempty=False)
不定期副业成功案例分享
OR
(仅在这种情况下),它会生成 SQLAND
。请参阅此页面以供参考:docs.djangoproject.com/en/dev/topics/db/queries/… 链接的优点是您可以混合使用exclude
和filter
来对复杂的查询条件进行建模。如果您想为真正的 SQL 建模OR
,您必须使用 Django Q 对象:docs.djangoproject.com/en/dev/topics/db/queries/…请编辑您的编辑以反映这一点,因为答案是严重误导。OR
来融合条件。我将编辑我的答案以澄清。NOT (A AND B)
等价于NOT A OR NOT B
。我认为这会让了解 SQL 但不熟悉 ORM 的新 Django 开发人员感到困惑。AND
转换为OR
的优势,因为您使用的是exclude
。在一般情况下,将链接视为THEN
(即exclude(A) THEN exclude(B)
)可能更正确。对上面的苛刻语言感到抱歉。您的回答确实很好,但我担心新开发人员过于笼统地接受您的回答。AND
和OR
呈现链接可能对从 SQL 背景进入 Django 的人有用。为了更深入地了解 Django,我认为文档做得比我做得更好。