django-ORM聚合函数

models.py的模型结构

# -*- coding: UTF-8 -*-
from django.db import models

class Author(models.Model):
    "作者模型"
    name = models.CharField(max_length=100)
    age = models.IntegerField()
    email = models.EmailField()

    class Meta:                 #该名称不能更改
        db_table = 'author'     #映射到数据库中为你自定义的名字,不设置这个默认为  包名_表名 映射到数据库


class Publisher(models.Model):
    "出版社模型"
    name = models.CharField(max_length=300)

    class Meta:
        db_table = 'publisher'

class Book(models.Model):
    "book模型"
    name = models.CharField(max_length=300)
    pages = models.IntegerField()
    price = models.FloatField()
    rating = models.FloatField()
    author = models.ForeignKey(Author,on_delete = models.CASCADE)
    publisher = models.ForeignKey(Publisher,on_delete=models.CASCADE)

    class Meta:
        db_table = 'book'


class BookOrder(models.Model):
    "图书订单模型"
    book = models.ForeignKey("Book",on_delete=models.CASCADE)
    price = models.FloatField()

    class Meta:
        db_table = 'book_order'

聚合函数

在views.py中

1.Avg : 求平均值

from django.shortcuts import render

from django.http import HttpResponse
from .models import Book
from django.db.models import Avg
from django.db import connection

def index(request):

    #获取book表下的价格(price字段)求平均价(平均值)
    result = Book.objects.aggregate(Avg('price'))  #Avg不能直接使用,需要使用aggregate包住
    print(result)
    print(connection.queries)    #可以查看被翻译成的sql语句,需要导入connection, 如果是queryset的话就可以使用query进行查询
    return HttpResponse('index')

这时候django就会自动给你创建一个别名,也就是 字段名__Avg,如果想要更给改的话

    #这时候django就会自动给你创建一个别名,也就是 字段名__Avg,如果想要更给改的话
    result = Book.objects.aggregate(price_avg=Avg('price'))  #Avg不能直接使用,需要使用aggregate包住

 

关于 aggregate 和 annotate 执行聚合函数的方法

例子 : 比如在求平均值
#1.aggregate 会将所有的值都会求出平均值(需要导入connection模块)

def index(request):
result = Book.objects(Avg('price'))  #Avg不能直接使用,需要使用aggregate包住详情看1的写法
print(result) #2.annotate 的话就会一个额外的 group by的分组操作,然后在求出各个组的平均值 #求各组的平均值 #将Book表下关联的BookOrder(使用的话就django默认设置的小写)的price字段中根据id的不同分组,然后在求出各个组的平均值 
books = Book.objects.annotata(price_avg=Avg('bookorder__price')) #将这个名称改了price_avg
for book in books: print('%s/%s'%(book.name,book.price_avg)) #price_avg在前面中已经更改了名称,不然就是默认的Avg

 Count 求每个表中有多少个id(也就是求总数)

from django.http import HttpResponse
from .models import Book,Author,BookOrder
from django.db.models import Avg,Count
from django.db import connection

    #1. Count 求每个表中有多少个id(也就是求总个数)
    # result = Book.objects.aggregate(book_nums =Count('id')) #book_nums自定义关键字参数,用户接收返回的数据
    # print(result)
    # print(connection.queries)   #查看转换成的sql语句

    #2.统计 Author 表下的有多少个不同的邮箱(也就是使用count统计相同的邮箱时只算一个,而不是全部统计)- 也就是distinct=True
    # result = Author.objects.aggregate(email_nums =Count('email',distinct=True)) #email_nums自定义关键字参数,用户接收返回的数据
    # print(result)
    # print(connection.queries)   #查看转换成的sql语句

    # 3. 使用 annotate 来统计每个分组的具体个数( 换言之也就是求每个组的销量等等这样的就可以使用这个方法 )
    books = Book.objects.annotate(book_nums=Count('bookorder__id'))   #bookorder__id 也就是bookorder表下的id字段
    for book in books:
        print('%s/%s'%(book.name,book.book_nums))   #使用了book_nums自定义参数那么这个表就自动增加了这个属性

Max  和 Min 获取指定对象(某字段)的最大值和最小值

from django.db.models import Avg,Count,Max,Min

    # 1. Max  和 Min 获取指定对象(某字段)的最大值和最小值
    result = Author.objects.aggregate(max=Max('sge'),min=Min('age')) #求这个Author表下的年龄字段的最大值和最小值,max和min是自定义的,用来接收值
    print(request)
    print(connection.queries)

#同理也可以使用annogate分组求每本书的最大值和最小值

Sum : 求某个字段的总和

from .models import Book,Author,BookOrder
from django.db.models import Avg,Count,Max,Min,Sum
#如果有漏的在自行导入,这只是一个大概例子

    #1.  求所有图书的销售总额,也就是所有卖出的书的总和
    # result = BookOrder.objects.aggregate(total=Sum('price'))    #求出BookOrder表下price字段的总和
    # print(request)
    # print(connection.queries)

    #2. 单独求出每一本书的销售总额(也就是关于三国演义这本书以及红楼梦这本书都各卖了多少钱)-需要到分组也就是 annotate
    books = BookOrder.objects.annotate(total=Sum('bookorder__id'))    #对bookorder__id这个字段做一个分组,然后求出每个分组的总额,然后使用自定义的total接收
    for book in books:
        print('%s%s'%(book.name,book.total))    #这个total是上面自定义的,自定义好厚就会为这个表添加上这个属性
    print(connection.queries)

    #3.求某一年的销售总额    -   这里只提供思路
    # ①先获取某一年的数据    表名.objects.filter 回顾之前的时间查询条件
    # ②再根据这些数据使用Sum求出来即可 ,使用正常的 aggregate  和 Sum


    #4.求每一种书在某一年的销售总额,也就是三国演义在2018年一共卖了多少钱,红楼梦在2015一共卖了多少钱
    #①先获取某一年的数据    表名.objects.filter 回顾之前的时间查询条件
    #②再根据这些数据使用Sum求出来即可 ,使用正常的 annotate  和 Sum

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

 

上一篇:django-聚合查询


下一篇:struts2中url标签