Django中的ORM操作之高级查询aggregate和annotate方法

今天毕设导师让我们拿着电脑去他办公室查看软件运行情况,然后让我加一些功能点和创新点,对博客内容进行数据挖掘,我想了想需要用到Django中的ORM操作之高级查询aggregate和annotate方法,那我们一起来学习一下吧。

aggregate()方法详解

aggregate的中文意思是聚合,源于SQL的聚合函数,Django的aggregate()方法作用是对一组值 (比如queryset的某个字段)进行统计计算,并以字典(Dict)格式返回统计计算结果。

django的aggregate方法支持的聚合操作有AVG / COUNT / MAX / MIN /SUM等。

annotate()方法详解

annotate的中文意思是注释,一个更好的理解是分组(Group By)。如果你想要对数据集先进行分组然后再进行某些聚合操作或排序时,需要使用annotate方法来实现。

与aggregate方法不同的是, annotate方法返回结果的不仅仅是含有统计结果的一个字典,而是包含有新增统计字段的查询集(queryset) 。

values或者values_list 放在annotate前面: 表示values或values_list是声明以什么字段分组,annotate执行分组。

values或者values_list放在annotate后面: annotate表示直接以当前表的pk执行分组, values或者values_list表示查询那些字段, 并且要将annotate里的聚合函数起别名, 在values或者values_list里面写其别名。

res = Publish.objects.values("name").annotate(in_price = Min("book__price"))

# 先以出版社的名字进行分组, 然后再使用聚合函数查询到每个出版社里面最便宜的书籍

aggregate和annotate应用场景

Django的aggregate和annotate方法属于高级查询方法,主要用于组合查询,可以大大提升数据库查询效率。

当你需要对查询集(queryset)的某些字段进行聚合操作时(比如Sum, Avg, Max),请使用aggregate方法。

如果你想要对数据集先进行分组(Group By)然后再进行某些聚合操作或排序时,请使用annotate方法。

Django中使用原生SQL

我一开始想图简化,就不想用sql,于是转战django中的orm,结果今天卡住了,想半天想不出来怎么转,又回到了sql,所以说啊,不要偷懒,不然会很痛苦的,血的教训!

那么问题来啦,Django中怎么使用原生Sql呢?

首先我现在的需求是,查询每一个类别的文章的总数目、总访问量、总评论量以及每一个作者的文章的总数目、总访问量、总评论量,对应的sql语句如下:

select category_id,count(id),sum(total_views),sum(comments) from tb_article group by category_id;
select author_id,count(id),sum(total_views),sum(comments) from tb_article group by author_id;

在这里插入图片描述
在这里插入图片描述
在Django中执行原生sql的操作如下:

# 查询各个类别、各个作者的文章的总数量、总访问量、总评论量
        try:
            # 创建连接
            conn = pymysql.connect(host='localhost', port=3306, user='root', passwd='wxm20010428', db='blog', charset='utf8')
            # 创建游标
            cursor = conn.cursor()
            # 原生sql语句  注意多条sql语句执行时的写法
            sql1 = 'select category_id,count(id),sum(total_views),sum(comments) from tb_article group by category_id;'
            sql2 = 'select author_id,count(id),sum(total_views),sum(comments) from tb_article group by author_id;'
            # 执行SQL
            cursor.execute(sql1)
            result1 = cursor.fetchall()
            for r1 in result1:
                print(r1)
            # 完成sql1的查询
            cursor.execute(sql2)
            result2 = cursor.fetchall()
            for r2 in result2:
                print(r2)
            # 完成sql2的查询
            cursor.close()
            conn.close()
        except Exception as e:
            print("Error%d: %s" % (e.args[0], e.args[1]))

执行结果如下:
在这里插入图片描述

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值