1

我的模型

class Machine(models.Model):
    machineName = models.CharField(verbose_name="Machine Name", max_length=20, blank=False, null=False)


class SalesReport(models.Model):
    machine = models.ForeignKey(Machine, on_delete=models.CASCADE, null=False, blank=False)
    deviceDate = models.CharField(max_length=200, null=True, blank=True)
    serverDate = models.DateTimeField(auto_now_add=True)
    totalPrice = models.FloatField()

我有 3 台机器,我想获得过去 7 天每台机器的总销售额。

我的查询是

from django.db.models import Sum, Value as V
from django.db.models.functions import Coalesce

SalesReport.objects.values("serverDate__date", "machine__machineName").annotate(
...                 sales=Coalesce(Sum("totalPrice"),V(0))).filter(
...                                                 serverDate__gte=week_start,
...                                                 serverDate__lte=week_end)

这给出了以下结果,

[{'serverDate__date': datetime.date(2020, 7, 22), 'machine__machineName': 'machine__1', 'sales': 15.0},
 {'serverDate__date': datetime.date(2020, 7, 28), 'machine__machineName': 'machine__1', 'sales': 145.0},
 {'serverDate__date': datetime.date(2020, 7, 28), 'machine__machineName': 'machine__2', 'sales': 270.0},
 {'serverDate__date': datetime.date(2020, 7, 28), 'machine__machineName': 'machine__3', 'sales': 255.0}]

我想要得到的是

[{'serverDate__date': datetime.date(2020, 7, 22), 'machine__machineName': 'machine__1', 'sales': 15.0},
{'serverDate__date': datetime.date(2020, 7, 22), 'machine__machineName': 'machine__2', 'sales': 0.0},
{'serverDate__date': datetime.date(2020, 7, 22), 'machine__machineName': 'machine__3', 'sales': 0.0},
{'serverDate__date': datetime.date(2020, 7, 28), 'machine__machineName': 'machine__1', 'sales': 145.0},
{'serverDate__date': datetime.date(2020, 7, 28), 'machine__machineName': 'machine__2', 'sales': 270.0},
{'serverDate__date': datetime.date(2020, 7, 28), 'machine__machineName': 'machine__3', 'sales': 255.0}]

我想用 Coalesce 来做,但我弄错了。

*我使用mysql作为数据库。一个数据库特定的查询也很好。

4

2 回答 2

0

由于它是更多 SQL 问题,因此我添加了更具体的答案

SELECT m.machineName, s.price
FROM machine m LEFT OUTER JOIN (
SELECT machine_id id, sum(totalPrice) price
FROM salesreport
WHERE serverDate BETWEEN DATE_SUB(curdate(), INTERVAL 1 WEEK) and curdate()
GROUP BY by machine_id) s on m.id = s.id

如果您希望 serverDate 作为输出,则必须应用聚合函数(Max,Min),因为它位于您的 SalesReport 表中。

这取决于 serverDate 代表什么。如果是您购买机器的日期,那么它应该在机器表中,并且可以直接从机器表中选择(并且WHERE BETWEEN子句必须存在子选择并且也适用于机器表)。如果它是 salesDate,那么它必须在 SalesReport 中,并且您必须对其应用聚合函数。即:你可以在一周内有 7 次约会......

SELECT m.machineName, s.MaxserverDate, s.price
FROM machine m LEFT OUTER JOIN (
SELECT machine_id id, max(serverDate) MaxserverDate, sum(totalPrice) price
FROM salesreport
WHERE serverDate BETWEEN DATE_SUB(curdate(), INTERVAL 1 WEEK) and curdate()
GROUP BY by machine_id) s on m.id = s.id
于 2020-07-29T08:51:10.923 回答
0

问题是您在某些日期没有任何销售。这更像是一个特定于数据库的问题,而不是一个 django ORM 问题。我建议在您的机器表上使用带有左外连接的原始 sql => 获取所有机器并在存在时列出销售。

machine = Machine.objects.raw('''
    SELECT machine.id, machine.name, sales.sid FROM app_machinelist as machine
    LEFT JOIN (select sales_id as sid
                 from app_sales
                where profile_id = {0}) sales
    ON sales.sid = machine.id
    ORDER BY machine.name ASC
    '''.format(myuser.id))

此示例有效,但出于安全原因,最好通过字典传递参数

machine = Machine.objects.raw(mysql, params)

在哪里

params = {'profile_id': pk, 'startdate': startdate, 'enddate': enddate}

mysql = '''
    SELECT machine.id, machine.name, sales.sid FROM app_machinelist as machine
    LEFT JOIN (select sales_id as sid
                 from app_sales
                where profile_id = %(profile_id)s) sales
    ON sales.sid = machine.id
    ORDER BY machine.name ASC
    '''
于 2020-07-28T16:39:16.110 回答