python django基础五 ORM多表操作
Posted 崽崽blog
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了python django基础五 ORM多表操作相关的知识,希望对你有一定的参考价值。
首先在创建表的时候看下分析一下
1.作者表和作者详细地址表 一对一关系 理论上谁都能当主表 把Author设置成主表 au=models.OneToOneField(to=\'AuthorDetail\',to_field=\'id\')
主表会多个au列 用于关联
2.书和出版社 一个出版社可以出好多书 一对多关系,哪个表数据多就把关键设置哪个表中 所以publisher=models.ForeignKey(to=\'Publish\',to_field=\'id\',on_delete=models.CASCADE)关联后publisher字段会变成publisher_id,有级联删除deltete
3.书和作者表 多对多关系 理论上谁都能当主表 把book设为主表 authors=models.ManyToManyField(to=\'Author\')这时候会产生第3个表app01_book_authors用于保存书id和作者id的对应的关系
创建表
from django.db import models #作者表 class Author(models.Model): id=models.AutoField(primary_key=True) name=models.CharField(max_length=30) age=models.IntegerField() #AuthorDetail 一对一表 关联 au=models.OneToOneField(to=\'AuthorDetail\',to_field=\'id\') #详细地址 class AuthorDetail(models.Model): id=models.AutoField(primary_key=True) address=models.CharField(max_length=40) tel=models.CharField(max_length=20) #出版社 class Publish(models.Model): id=models.AutoField(primary_key=True) name=models.CharField(max_length=20) addr=models.CharField(max_length=40) #书 class Book(models.Model): id=models.AutoField(primary_key=True) title=models.CharField(max_length=20) #书和出版社 出版社一对多 publisher=models.ForeignKey(to=\'Publish\',to_field=\'id\',on_delete=models.CASCADE) #多对多 书跟作者 authors=models.ManyToManyField(to=\'Author\')
python manage.py makemigrations
python manage.py migrate
添加数据
一对一
先添加地址表 models.AuthorDetail.objects.create( id=2,address=\'杭州\',tel=777 ) #如果写对应另一个表id 需要写au_id字段
models.Author.objects.create( name=\'张无忌\',age=18,au_id=1 )
一对多
方法一 查询出版社id=1 pub_obj=models.Publish.objects.filter(id=1)[0] models.Book.objects.create(title=\'西游记\',publisher=pub_obj) #publisher是设计字段
publisher_id是生产的 方法二 models.Book.objects.create( title=\'三国演义\', publisher_id=1 )
book_obj.authors.add(*[1,2])#添加多参数
多对多
# 多对多插入 book_obj = models.Book.objects.get(id=2) lzs = models.Author.objects.get(id=2) ##第二种方式 # book_obj.authors.add(*[1,2]) book_obj.authors.add(2, 2)
删除:
models.Author.objects.filter(id=1).delete() #删除作者的不会删除详细信息 models.Book.objects.filter(id=1).delete() #删除某一本书的时候会删除书相关的第三张表 book_obj.authors.clear()#全部清空 book_obj.authors.remove(\'2\')
应用地址
models.Book.objects.get(id=n).delete()
更新
models.Author.objects.filter(id=1).update(au_id=5) au=authordetail对象 book_obj = models.Book.objects.filter(id=1)[0] models.Book.objects.filter(id=1).update(title=\'xxxxx\',publisher_id=4) book_obj.authors.set([\'3\',\'4\']) #应用地址 models.Book.objects.filter(id=n).update(**data) one_book.authors.set(authors_lsit)
查询数据
一对一查询
##正向一对一查询 zuozhe=models.Author.objects.get(name=\'张无忌\') print(zuozhe.au.address)#通过 au映射 作者信息 找到地址 ##反向一对一查询 fanzuozhe=models.AuthorDetail.objects.get(address=\'台湾\') print(fanzuozhe.author.name)#author是作者名称
一对多查询
# 一对多查询 shu=models.Book.objects.get(title=\'三国演义\') print(shu.publisher.name)#通过书名 publisher正向查询出版社name #反向查询 fanshu=models.Publish.objects.get(name=\'18期出版\') print(fanshu.book_set.all().values())
多对多查询
##多对多查询 shu=models.Book.objects.get(title=\'三国演义\') print(shu.authors.all().values(\'name\')) fanshu=models.Author.objects.get(name=\'张无忌\') print(fanshu.book_set.all().values())
双__线查法
正向----->字段名__属性
反向----->表名__属性
#查询18期出版 出版了哪些书 ret=models.Book.objects.filter(publisher__name=\'18期出版\').values(\'title\') #先找book表到出版社属性中的name values值等于title ret=models.Publish.objects.filter(name=\'18期出版\').values(\'book__title\') #先找到出版社18期values等于book表的title #天龙八部这本书是哪些作者写的 ret=models.Book.objects.filter(title=\'天龙八部\').values(\'authors__name\') # print(ret)#<QuerySet [{\'authors__name\': \'张无忌\'}, {\'authors__name\': \'赵敏\'}]> ret=models.Author.objects.filter(book__title=\'天龙八部\').values(\'name\') # print(ret)#<QuerySet [{\'name\': \'张无忌\'}, {\'name\': \'赵敏\'}]>
#查询18期出版 出版过的所有书籍的名字以及作者的姓名 找到作者 因为作者和出版社没关系 所以要通过authors ret=models.Publish.objects.filter(name=\'18期出版\').values(\'book__title\',\'book__authors__name\') # print(ret) #通过publisher 找到18期出版 values等于title 然后通过authors查作者名称 ret=models.Book.objects.filter(publisher__name=\'18期出版\').values(\'title\',\'authors__name\') # print(ret)
#手机号以2开头的作者出版过的所有书籍名称以及出版社名称 #书表通过authors作者,au找到tel 6开头的 书名 通过publisher查出版社name ret=models.Book.objects.filter(authors__au__tel__startswith=\'6\').values(\'title\',\'publisher__name\') print(ret)
聚合查询
聚合是同aggreate(*args,**kwarges),通过querySet进行计算 主要有Sum,Avg,Max,Min
from django.db.models import Sum,Avg,Max,Min,Count
Sum 总和
ret=models.Author.objects.all().aggregate(Sum("age")) print(ret) 85
Avg 平均值
ret=models.Author.objects.all().aggregate(Avg("age")) print(ret)#{\'age__avg\': 28.3333}
还可以对结果进行别名
ret=models.Author.objects.all().aggregate(age=Avg("age"))
print(ret)#{\'age\': 28.3333}
Max和Min
ret=models.Author.objects.all().aggregate(Max(\'age\'),Min(\'age\'),age=Avg("age")) print(ret)#{\'age\': 28.3333, \'age__max\': 55.0, \'age__min\': 10.0}
Count(计数)
ret=models.Author.objects.all().aggregate(Count(\'name\')) print(ret)#{\'name__count\': 3}
分组查询
将查询结果按照某个字段或字段分组进行分组,字段中相等的为一组
annotate()为调用的QuerySet中每一个对象都生成一个独立的统计值
单表分组查询
#统计每个部门的人数 ret=models.Author.objects.values(\'dep\').annotate(c=Count(\'id\')) print(ret)#<QuerySet [{\'dep\': \'人力\', \'c\': 2}, {\'dep\': \'人事\', \'c\': 1}, {\'dep\': \'财务\', \'c\': 1}]> #统计每个部门的平局工资 ret=models.Author.objects.values(\'dep\').annotate(avg=Avg(\'salary\')) print(ret)
values(\'dep\')表示以dep字段进行分组
annotate(按每个组的id字段进行统计)
多表分组查询
#每个出版社出了多少书 ret=models.Book.objects.values(\'publisher__id\',\'publisher__name\').annotate(c=Count(\'id\')) print(ret)#<QuerySet [{\'publisher__id\': 1, \'publisher__name\': \'太阳出版社\', \'c\': 2}, {\'publisher__id\': 2, \'publisher__name\': \'月亮出版社\', \'c\': 1}]> #每个作者出了多少书 ret=models.Book.objects.values(\'authors__name\').annotate(c=Count(\'title\')) print(ret)#<QuerySet [{\'authors__name\': \'张无忌33\', \'c\': 1}, {\'authors__name\': \'周芷若\', \'c\': 1}, {\'authors__name\': \'赵敏\', \'c\': 2}]> #查询每一本书的名称及作者个数 对id进行分组,按authors__name ret=models.Book.objects.annotate(count=Count("authors__name")).values(\'title\',\'count\') print(ret)#<QuerySet [{\'title\': \'倚天屠龙记\', \'count\': 1}, {\'title\': \'天龙八部\', \'count\': 2}, {\'title\': \'月亮代表我的心\', \'count\': 1}]> #查询大于一个作者的书籍名称 ret=models.Book.objects.filter(count__gt=1).values(\'title\',\'count\') print(ret)#<QuerySet [{\'title\': \'天龙八部\', \'count\': 2}]> # 查询每个出版色最便宜的书籍名称 ret = models.Publish.objects.annotate(MinPrice=Min("book__price")).values_list("name", "MinPrice") print(ret)#<QuerySet [(\'太阳出版社\', Decimal(\'100.00\')), (\'月亮出版社\', Decimal(\'300.00\'))]> #统计每一本以py开头的书籍的作者个数 ret=models.Book.objects.filter(title__startswith="天").annotate(num_authors=Count(\'authors\')).values_list("title","num_authors") print(ret)#<QuerySet [(\'天龙八部\', 2), (\'天天开心\', 1)]> #查询每个作者的姓名以及出版的书的最高价格 ret=models.Author.objects.values(\'name\').annotate(Max(\'book__price\')) ret=models.Author.objects.annotate(jiage=Max(\'book__price\')).values(\'name\',\'jiage\') #2 查询作者id大于2作者的姓名以及出版的书的最高价格 ret = models.Author.objects.values(\'name\').filter(id__gt=2).annotate(Max(\'book__price\')) #3 查询作者id大于2或者作者年龄大于等于20岁的女作者的姓名以及出版的书的最高价格 ret = models.Author.objects.filter(Q(id__gt=2)|Q(age__gte=20),sex=\'female\').values(\'name\').annotate(Max(\'book__price\')) #4 查询每个作者出版的书的最高价格 的平均值 ret = models.Author.objects.values(\'name\').annotate(aa=Max(\'book__price\')).aggregate(Avg(\'aa\'))
多表分组查询
F对象之间的加减乘除
from django.db.models import F
#把Book的价格字段 批量+30 models.Book.objects.all().update(price=F("price")+ 30) #查询年纪小于 工资的name __gt 表示大于 models.Author.objects.filter(age__lt=F(\'salary\')).values("name")
Q对象的复杂查询
对象可以使用&(与) 、|(或)、~(非)
from django.db.models import Q
#分别查询作者周芷若、赵敏出版过的书籍名称 ret=models.Book.objects.filter(Q(authors__name="周芷若")|Q(authors__name="赵敏")).values("title")#如果2个人合出的书,会出现2个title
等同于下面的SQL WHERE 子句: WHERE name ="周芷若" OR name ="赵敏"
ret=models.Book.objects.filter(~Q(authors__name="赵敏")).values("title") print(ret)#查询非赵敏
ret=models.Book.objects.filter(Q(authors__name="周芷若") & ~Q(authors__salary=6000)).values_list("title")
print(ret)
#分别查询作者周芷若、赵敏出版过的书籍名称 ret=models.Book.objects.filter(authors__name="周芷若").filter(authors__name="赵敏").values_list("title") print(ret)#<QuerySet [(\'天龙八部\',)]>
事物
#正常只有第一条被成功插入 from app01 import models from django.db import transaction models.Publish.objects.create(name=\'青蛙出版社1\', addr=\'井里1\') with transaction.atomic(): # ret = models.Publish.objects.all().values(\'name\') models.Publish.objects.create(name=\'青蛙出版社2\',addr=\'井里2\') models.Publish.objects.create(id=\'青蛙出版社3\',addr=\'井里3\')
以上是关于python django基础五 ORM多表操作的主要内容,如果未能解决你的问题,请参考以下文章