博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
django orm中利用annotate进行group by
阅读量:6263 次
发布时间:2019-06-22

本文共 2243 字,大约阅读时间需要 7 分钟。

hot3.png

用法

之前的orm的group by方法在django 1.8 中已经不能使用,需要利用annotate来实现

  • 示例1

第一个values用来选中需要用来group by的字段(此处group by user_id),之后紧跟annotate来分组并聚合需要的字段(需要每个user_id对应的question_id的数量和catalog_id的最小值),之后再values来实际查询需要的字段(原user_id和聚合后的字段的别名)

  1. 第一个values用来指定用来group by的字段,里面必须是Count、Min等等聚合函数(例如用F("user_id")取别名是不行的),不需要最终查询的就不必聚合了

  2. 第二个values用来指定实际select的字段,只能指定annotate后的字段名(以此处为例:user_id是用来分组的字段,可以直接取,而其他字段必须聚合并使用聚合后的别名,qid和cid,假如原表还有个字段status,annotate中没有聚合此字段,所以最后value不能查询该字段)

q = PxbNCEUserQuest.objects.filter(user_id=335).values("user_id").annotate(qid=Min("question_id"), cid=Min("catalog_id")).values("user_id", "qid", "cid")print qprint q.query# 输出[{'qid': 22, 'user_id': 335L, 'cid': 17}]SELECT `pxb_nce_user_quest`.`user_id`, MIN(`pxb_nce_user_quest`.`question_id`) AS `qid`, MIN(`pxb_nce_user_quest`.`catalog_id`) AS `cid` FROM `pxb_nce_user_quest` WHERE `pxb_nce_user_quest`.`user_id` = 335 GROUP BY `pxb_nce_user_quest`.`user_id` ORDER BY NULL
  • 示例2
  1. 与示例1一样,但是此处第一个annotate用来分组字段,第二个annotate用来单独别名其他字段
q = PxbNCEUserQuest.objects.filter(user_id=335).values("user_id").annotate(qid=Min("question_id"), cid=Min("catalog_id")).annotate(uid=F("user_id")).values("uid", "qid", "cid")print qprint q.query# 输出:[{'qid': 22, 'uid': 335L, 'cid': 17}]SELECT MIN(`pxb_nce_user_quest`.`question_id`) AS `qid`, MIN(`pxb_nce_user_quest`.`catalog_id`) AS `cid`, `pxb_nce_user_quest`.`user_id` AS `uid` FROM `pxb_nce_user_quest` WHERE `pxb_nce_user_quest`.`user_id` = 335 GROUP BY `pxb_nce_user_quest`.`user_id` ORDER BY NULL

举例:

  • SomeModel.objects.annotate(Count('somecol'))

GROUP BY: 所有字段

  • SomeModel.objects.values('name').annotate(Count('somecol'))

GROUP BY: name字段,聚合somecol

  • SomeModel.objects.annotate(Count('somecol')).values('name')

GROUP BY: 所有字段,查询name

  • SomeModel.objects.values('name', 'pk').annotate(Count('somecol')).values('pk')

GROUP BY: name, pk字段,查询pk字段

  • SomeModel.objects.values('name').annotate(Count('somecol')).values('pk')

GROUP BY: name, pk字段,查询pk字段

关联知识:

刚开始上面的查询方法可能比较难理,但是对比原生sql语句的group by方法就会发现类似原理

  • 老版本mysql中

select a, b from t group by a会正常工作,b字段会自动取第一条,等于是隐式聚合了

  • 新版本mysql中以上语句不能工作,因为默认启用严格模式sql_mode=ONLY_FULL_GROUP_BY,正确方法是:

select a,max(b) as b from t group by,即需要显示的聚合所有查询的字段

对比新版mysql语法会发现跟orm中查询方法很类似

转载于:https://my.oschina.net/watcher/blog/775258

你可能感兴趣的文章
日请求亿级的 QQ 会员 AMS 平台 PHP7 升级实践
查看>>
【Coursera】Security Introduction -Eighth Week(2)
查看>>
Vue系列:如何将百度地图包装成Vue的组件
查看>>
Charles 从入门到精通
查看>>
MVC5 + EF6 简单示例
查看>>
Mysql Innodb存储引擎 insert 死锁分析
查看>>
好的用户界面-界面设计的一些技巧
查看>>
全端开发必备!10个最好的 Node.js MVC 框架
查看>>
初始Knockout
查看>>
HADOOP 2.6 INSTALLING ON UBUNTU 14.04 (hadoop 2.6 部署到ubuntu 14.04上面)
查看>>
OSSIM架构与组成综述
查看>>
用ASP.NET Core 2.0 建立规范的 REST API -- 预备知识 (2) + 准备项目
查看>>
数据分析:基于Python的自定义文件格式转换系统
查看>>
如何重置Sitecore CMS中的管理员密码
查看>>
[SilverLight]DataGrid实现批量输入(like Excel)(补充)
查看>>
PHP 杂谈《重构-改善既有代码的设计》之三 重新组织数据
查看>>
NSBundle介绍
查看>>
POJ1811_Prime Test【Miller Rabin素数測试】【Pollar Rho整数分解】
查看>>
ConnectString中enlist设置的含义
查看>>
潜移默化学会WPF(企业经验篇)--Log4Net(二)
查看>>