Django的ORM如何执行group by 语句

问题描述:

使用Django的ORM建立了如下Model:

class Book(models.Model):
    name = models.CharField(max_length=300)
    pages = models.IntegerField()
    price = models.DecimalField(max_digits=10, decimal_places=2)
    authors = models.CharField(max_length=300)
    publisher = models.CharField(max_length300)
    pub_date = models.DateField()

 

现在想查询以字段authors分组,查出每个作者所著所有书的总页书,用SQL语句可以很容易的完成此任务:

SELECT *, SUM(pages) AS total_pages
FROM Book
GROUP BY authors;

 那么如何用Django自带的ORM接口实现呢?

from django.db.models import Sum

result = Book.objects.values("authors").annotate(total_pages=Sum("pages")).all()

 

注意事项:

如果你在定义Book模型时指定的默认的排序字段,如下:

class Book(models.Model):

    ...同上...
    
    class Meta:
        ordering = ["name"]

 

 那么,上述result的查询语句会得到错误的结果, 其所执行的SQL语句如下:

SELECT *, SUM(pages) AS total_pages
FROM Book
GROUP BY authors, name
ORDER BY name;

也就是说默认的排序字段(在这里是name)也会自动加入到GROUP BY子句中. 要避免这样的情况,只需在查询时清空默认的order by排序条件即可:

from django.db.models import Sum

result = Book.objects.values("authors").annotate(total_pages=Sum("pages")).order_by()

 

ps:你可以通过以下方法查看查询集的对应SQL语句

raw_sql = result.query
print(raw_sql)

 

或者安装django-extensions插件,然后使用python manage.py shell_plus --print-sql命令进入Shell,在此Shell中你通过Django对数据库执行的所有操作都会自动打印出其对应的SQL语句.

除了上述方法,还有另外一种:

book_list = Book.objects.all()
book_list.query.group_by = ['authors']



作者:给我二两面
链接:https://www.jianshu.com/p/9ac4ca98b8e4
来源:简书
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

 

 

 

 

  • 1
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值