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')

 

posted @ 2019-03-01 00:01  崽崽1573  阅读(355)  评论(0编辑  收藏  举报