条件表达式

    在后续的例子中,我们将使用以下模型:

    class When(condition=None, then=None, \*lookups*)

    When() 对象用于封装一个条件及其结果,以便在条件表达式中使用。使用 When() 对象类似于使用 filter() 方法。可以使用 、 Q 对象或 对象来指定条件,这些对象的 output_field 是 BooleanField。结果是用 then 关键字提供的。

    Changed in Django 4.0:

    Support for lookup expressions was added.

    一些例子:

    1. >>> from django.db.models import F, Q, When
    2. >>> # String arguments refer to fields; the following two examples are equivalent:
    3. >>> When(account_type=Client.GOLD, then='name')
    4. >>> When(account_type=Client.GOLD, then=F('name'))
    5. >>> # You can use field lookups in the condition
    6. >>> from datetime import date
    7. >>> When(registered_on__gt=date(2014, 1, 1),
    8. ... registered_on__lt=date(2015, 1, 1),
    9. ... then='account_type')
    10. >>> # Complex conditions can be created using Q objects
    11. >>> When(Q(name__startswith="John") | Q(name__startswith="Paul"),
    12. ... then='name')
    13. >>> # Condition can be created using boolean expressions.
    14. >>> from django.db.models import Exists, OuterRef
    15. >>> non_unique_account_type = Client.objects.filter(
    16. ... account_type=OuterRef('account_type'),
    17. ... ).exclude(pk=OuterRef('pk')).values('pk')
    18. >>> When(Exists(non_unique_account_type), then=Value('non unique'))
    19. >>> # Condition can be created using lookup expressions.
    20. >>> from django.db.models.lookups import GreaterThan, LessThan
    21. >>> When(
    22. ... GreaterThan(F('registered_on'), date(2014, 1, 1)) &
    23. ... LessThan(F('registered_on'), date(2015, 1, 1)),
    24. ... then='account_type',
    25. ... )

    请记住,每个值都可以是一个表达式。

    注解

    1. >>> When(then__exact=0, then=1)
    2. >>> When(Q(then=0), then=1)

    Changed in Django 3.2:

    增加了对在 lookups 中使用 condition 参数的支持。

    class Case(\cases, **extra*)

    Case() 表达式就像 Python 中的 … elif… 语句。在提供的 When() 对象中的每个 condition 按顺序执行,直到执行出一个对的值。从匹配的 When() 对象中返回 result 表达式。

    一个例子:

    1. >>>
    2. >>> from datetime import date, timedelta
    3. >>> Client.objects.create(
    4. ... account_type=Client.REGULAR,
    5. ... registered_on=date.today() - timedelta(days=36))
    6. >>> Client.objects.create(
    7. ... name='James Smith',
    8. ... account_type=Client.GOLD,
    9. ... registered_on=date.today() - timedelta(days=5))
    10. >>> Client.objects.create(
    11. ... name='Jack Black',
    12. ... account_type=Client.PLATINUM,
    13. ... registered_on=date.today() - timedelta(days=10 * 365))
    14. >>> # Get the discount for each Client based on the account type
    15. >>> Client.objects.annotate(
    16. ... discount=Case(
    17. ... When(account_type=Client.GOLD, then=Value('5%')),
    18. ... When(account_type=Client.PLATINUM, then=Value('10%')),
    19. ... default=Value('0%'),
    20. ... ),
    21. ... ).values_list('name', 'discount')
    22. <QuerySet [('Jane Doe', '0%'), ('James Smith', '5%'), ('Jack Black', '10%')]>

    Case() 接受任意数量的 When() 对象作为单个参数。其他选项是通过关键字参数提供的。如果没有一个条件的值是 TRUE,那么将返回用 default 关键字参数给出的表达式。如果没有提供 default 参数,则使用 None

    如果我们想改变之前的查询,根据 Client 与我们合作的时间长短来获取折扣,我们可以使用查找:

    注解

    Case() 也可以在 filter() 子句中使用。例如,要查找一个月前注册的黄金客户和一年前注册的白金客户:

    1. >>> a_month_ago = date.today() - timedelta(days=30)
    2. >>> a_year_ago = date.today() - timedelta(days=365)
    3. >>> Client.objects.filter(
    4. ... registered_on__lte=Case(
    5. ... When(account_type=Client.GOLD, then=a_month_ago),
    6. ... When(account_type=Client.PLATINUM, then=a_year_ago),
    7. ... ),
    8. ... ).values_list('name', 'account_type')
    9. <QuerySet [('Jack Black', 'P')]>

    高级查询

    条件表达式可用于注解、聚合、过滤器、查找和更新中。它们还可以与其他表达式组合和嵌套。这使你可以进行强大的条件查询。

    比方说,我们想改变客户的 account_type 以匹配他们的注册日期。我们可以使用一个条件表达式和 方法来实现这个目标:

    1. >>> a_month_ago = date.today() - timedelta(days=30)
    2. >>> a_year_ago = date.today() - timedelta(days=365)
    3. >>> # Update the account_type for each Client from the registration date
    4. >>> Client.objects.update(
    5. ... account_type=Case(
    6. ... When(registered_on__lte=a_year_ago,
    7. ... then=Value(Client.PLATINUM)),
    8. ... then=Value(Client.GOLD)),
    9. ... default=Value(Client.REGULAR)
    10. ... )
    11. >>> Client.objects.values_list('name', 'account_type')
    12. <QuerySet [('Jane Doe', 'G'), ('James Smith', 'R'), ('Jack Black', 'P')]>

    如果我们想知道每个 account_type 有多少个客户?我们可以使用 聚合函数filter 参数来实现:

    1. >>> # Create some more Clients first so we can have something to count
    2. >>> Client.objects.create(
    3. ... name='Jean Grey',
    4. ... account_type=Client.REGULAR,
    5. ... registered_on=date.today())
    6. >>> Client.objects.create(
    7. ... name='James Bond',
    8. ... account_type=Client.PLATINUM,
    9. ... registered_on=date.today())
    10. >>> Client.objects.create(
    11. ... name='Jane Porter',
    12. ... account_type=Client.PLATINUM,
    13. ... registered_on=date.today())
    14. >>> # Get counts for each value of account_type
    15. >>> from django.db.models import Count
    16. >>> Client.objects.aggregate(
    17. ... regular=Count('pk', filter=Q(account_type=Client.REGULAR)),
    18. ... gold=Count('pk', filter=Q(account_type=Client.GOLD)),
    19. ... platinum=Count('pk', filter=Q(account_type=Client.PLATINUM)),
    20. ... )
    21. {'regular': 2, 'gold': 1, 'platinum': 3}

    在支持 SQL 2003 FILTER WHERE 语法的数据库上,这个聚合产生一个查询。

    在其他数据库中,这是用 CASE 语句模拟的:

    1. SELECT count(CASE WHEN account_type=1 THEN id ELSE null) as regular,
    2. count(CASE WHEN account_type=2 THEN id ELSE null) as gold,
    3. count(CASE WHEN account_type=3 THEN id ELSE null) as platinum
    4. FROM clients;

    这两条 SQL 语句在功能上是等同的,但更明确的 FILTER 可能表现得更好。

    当条件表达式返回一个布尔值时,可以直接在过滤器中使用它。这意味着它不会被添加到 SELECT 列中,但你仍然可以使用它来过滤结果:

    1. >>> non_unique_account_type = Client.objects.filter(
    2. ... account_type=OuterRef('account_type'),
    3. ... ).exclude(pk=OuterRef('pk')).values('pk')
    4. >>> Client.objects.filter(~Exists(non_unique_account_type))
    1. SELECT ...
    2. FROM client c0
    3. WHERE NOT EXISTS (
    4. SELECT c1.id
    5. FROM client c1
    6. WHERE c1.account_type = c0.account_type AND NOT c1.id = c0.id