Django的orm练习---多表查询

Posted mr-wangxd

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Django的orm练习---多表查询相关的知识,希望对你有一定的参考价值。

 表关系如下

技术分享图片

表结构 : 

技术分享图片
from django.db import models

# Create your models here.


# 多对多----->>>老师和班级
# 一对多----->>>学生和班级  : 一个班级可以有多个学生----学生设置主键
#               老师和课程  : 一个老师可以教多门课程----课程设置主键
#
# 一对一---->>>>班级和年级 : 一个班级对应一个年级
#
# 成绩表----学生 : 一对多.  一个学生多个成绩 /
# 成绩表----课程 : 一对多

# 学生
class Student(models.Model):
    sid=models.AutoField(primary_key=True)
    sname=models.CharField(max_length=32)
    gender=models.CharField(max_length=32)
    class_id=models.ForeignKey(to="Class",on_delete=models.CASCADE)

# 班级
class Class(models.Model):
    cid=models.AutoField(primary_key=True)
    caption=models.CharField(max_length=32)
    grade=models.ForeignKey(to="Class_grade",on_delete=models.CASCADE)
    teachers=models.ManyToManyField(to="Teacher")

# 年级
class Class_grade(models.Model):
    gid=models.AutoField(primary_key=True)
    gname=models.CharField(max_length=32)

# 课程
class Course(models.Model):
    cid=models.AutoField(primary_key=True)
    cname=models.CharField(max_length=32)
    teacher=models.ForeignKey(to="Teacher",on_delete=models.CASCADE)

# 老师
class Teacher(models.Model):
    tid=models.AutoField(primary_key=True)
    tname=models.CharField(max_length=32)


#成绩
class Score(models.Model):
    sid=models.AutoField(primary_key=True)
    student=models.ForeignKey(to="Student",on_delete=models.CASCADE)
    course=models.ForeignKey(to="Course",on_delete=models.CASCADE)
    score=models.IntegerField()
技术分享图片

 

习题 : 

技术分享图片
# 1、自行创建测试数据;
# 2、查询学生总人数;
# 3、查询“生物”课程和“物理”课程成绩都及格的学生id和姓名;
# 4、查询每个年级的班级数,取出班级数最多的前三个年级;
# 5、查询平均成绩最高的学生的id和姓名以及平均成绩;
# 6、查询每个年级的学生人数;
# 7、查询每位学生的学号,姓名, 平均成绩;
# 8、查询学生编号为“2”的学生的姓名、该学生成绩最高的课程名及分数;
# 9、查询姓“李”的老师的个数和所带班级数;
# 10、查询班级数小于5的年级id和年级名;
# 11、查询教过课程超过2门的老师的id和姓名;
# 12、查询学过编号“1”课程和编号“2”课程的同学的学号、姓名;
# 13、查询所带班级数最多的老师id和姓名;
# 14、查询有课程成绩小于60分的同学的学号、姓名;
# 15、查询男生、女生的人数,按倒序排列;
# 16、 查询各个课程及相应的选修人数;
# 17、 查询同时选修了物理课和生物课的学生id和姓名;
# 18、 检索“3”课程分数小于60,按分数降序排列的同学学号;
# 19、 查询每门课程的平均成绩,结果按平均成绩升序排列,平均成绩相同时,按课程号降序排列;
# 20、 查询各科成绩最高和最低的分:以如下形式显示:课程ID,最高分,最低分;
技术分享图片

 

答案 : 

技术分享图片
from django.shortcuts import render,HttpResponse,redirect

from app01.models import Class,Course,Teacher,Student,Score,Class_grade

from django.db.models import F, Q

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

# Create your views here.

# values.annotate() : 按字段分组
# annotate() : 按id,name...分组,属于一列是一组

def query(request):

    # 2 . 查询学生总人数
    ret=Student.objects.count()
    print("------>",ret)   # {‘c‘: 3}

    # 3 . 查询“生物”课程和“物理”课程成绩都及格的学生id和姓名;
    ret = Student.objects.filter(score__course__cid=3,score__score__gt=59).filter(score__course__cid=1,score__score__gt=59).values("sname", "sid")
    print("----------------", ret)

    # 先去筛选选课在生物和物理之间的,并且创建大于60的学生id,在分组查看选课数,然后筛选选课数等于2的
   ret=Score.objects.filter(course__cname__in=["生物","物理"],score__gt=60).values("student__tid").annotate(c=Count("course")).filter(c=2)


    # 4.查询每个年级的班级数,取出班级数最多的前三个年级;
    ret=Score.objects.values("student").annotate(avg_score=Avg("score")).order_by("-avg_score").values("student__sname","student__pk","avg_score")[0]
    print(ret)

    ret=Class.objects.values("grade__gname").annotate(c=Count("caption")).order_by("-c")[:3]
    print("------>",ret)

    # 5.查询平均成绩最高的学生的id和姓名以及平均成绩;
    ret=Grade.objects.annotate(c=Count("klass__student__pk")).values("gname","c")
    print(ret)

    ret=Student.objects.values("sid","sname").annotate(scoreAvg = Avg("score__score")).order_by("-scoreAvg")[:1]
    print("------>", ret)

    # 6.查询每个年级的学生人数;
    ret=Student.objects.values("sid","sname").annotate(avg_score=Avg("score__score"))
    print(ret)

    ret=Student.objects.annotate(avg_score=Avg("score__score")).values("sid","sname","avg_score")

    ret=Class.objects.values("grade__gname").annotate(c=Count("student"))
    print("-------->",ret)

    # 7 . 查询每位学生的学号,姓名,平均成绩;
    ret=Student.objects.values("sid","sname").annotate(scoreAvg=Avg("score__score"))
    print("-------->", ret)

    ret=Score.objects.filter(student__pk=2).order_by("-score").values("student__sname","course__cname","score")[0]
    print(ret)

    # 8、查询学生编号为“2”的学生的姓名、该学生成绩最高的课程名及分数;
    ret=Student.objects.filter(sid="2").annotate(scoreMax=Max("score__score")).order_by(‘-scoreMax‘)[0:1].values("sname","score__course__cname","scoreMax")
    ret2=Student.objects.filter(sid="2").values("score__course").order_by("-score__score").values("sname","score__course__cname","score__score")[:1]
    print("-------->", ret)
    print("-------->", ret2)

    # 9、查询每一个姓“李”的老师所带班级数;;
    ret=Teacher.objects.filter(tname__istartswith="李").annotate(c=Count("classes")).values("tname","c")
    print(ret)


     # 10 . 查询班级数小于5的年级id和年级名;
    ret=Class_grade.objects.annotate(c=Count("class")).filter(c__lt=2).values("gid","gname")
    print("--------", ret)

    ret=Grade.objects.annotate(c=Count("klass")).filter(c__lt=5).values("pk","gname")


    # 11 . 查询教过课程超过2门的老师的id和姓名;
    ret=Teacher.objects.annotate(c=Count("course")).filter(c__gt=2).values_list("tid","tname")
    print("--------", ret)


    # 12 . 查询学过编号“1”课程和编号“2”课程的同学的学号、姓名;      ????????
    ret=Student.objects.filter(score__course__cid=1).filter(score__course__cid=2).values("sid","sname")
    print("-------",ret)


    # 13 . 查询所带班级数最多的老师id和姓名;
    ret=Teacher.objects.annotate(c=Count("class__cid")).order_by("-c").values("tid","tname","c")[0]
    print(">>>>>>>>>>",ret)


    # 14 . 查询有课程成绩小于60分的同学的学号、姓名;
    ret=Student.objects.filter(score__score__lt=60).values("sid","sname","score__sid").distinct()     #去重
    print("-------",ret)

    ret=Score.objects.filter(score__lt=60).values("student__sname","student__pk").distinct()
    print(ret)


    # 15 . 查询男生、女生的人数,按倒序排列;
    ret=Student.objects.values("gender").annotate(c=Count("gender")).order_by("-c")
    print("-------",ret)

    # 16 . 查询各个课程及相应的选修人数;
    ret=Course.objects.annotate(c=Count("score__student")).values("cname","c")
    print("-------", ret)
    ret=Score.objects.values("course").annotate(c=Count(1)).values("course__cname","c")
    print(ret)

    # 17 . 查询同时选修了物理课和生物课的学生id和姓名;
    ret=Student.objects.filter(score__course__cname="物理").filter(score__course__cname="生物").values("sid","sname")
    print("------->",ret)


    # 18 . 检索“3”课程分数小于60,按分数降序排列的同学学号;
    ret=Student.objects.filter(Q(score__course__cid=3),Q(score__score__lt=60)).order_by("-score__score").values("sid")   
    ret=Score.objects.filter(course__cid=3).filter(score__lt=60).order_by("-score").values("student__sid")
    print("------->", ret)

    ret=Score.objects.filter(course_id=3,score__lt=60).order_by("-score").values("student_id")


    # 19 . 查询每门课程的平均成绩,结果按平均成绩升序排列,平均成绩相同时,按课程号降序排列;
    ret=Score.objects.values("course_id").annotate(scoreAvg=Avg("score")).order_by("scoreAvg","-course_id")     #可以不用跨表  course_id
    print("------->", ret)


    # 20 . 查询各科成绩最高和最低的分:以如下形式显示:课程ID,最高分,最低分;
    ret=Score.objects.values("course__cid").annotate(scoreMax=Max("score"),scoreMin=Min("score"))
    print(">>>>>>>>>>>",ret)
   ret=Course.objects.annotate(max_score=Max("score__score"),min_score=Min("score__score")).values("pk","max_score","min_score")



    return HttpResponse("ok")
技术分享图片

以上是关于Django的orm练习---多表查询的主要内容,如果未能解决你的问题,请参考以下文章

python 之 Django框架(orm单表查询orm多表查询聚合查询分组查询F查询 Q查询事务Django ORM执行原生SQL)

[Django框架之ORM操作:多表查询,聚合查询分组查询F查询Q查询choices参数]

Django-ORM-多表操作

八Django的orm之多表操作

Django--ORM 多表查询

django ORM model filter 条件过滤,及多表连接查询反向查询,某字段的distinct