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)