9.django之多表操作

创建模型

表与表之间的关系

   一对一     一对多   多对多

假设有作者模型,作者详细模型,出版社模型,书籍模型

作者模型与作者详细模型是一对一关系

出版社模型与书籍模型是一对多关系

作者模型与出版社模型是多对多关系

模型如下

from django.db import models


# 作者模型
class Author(models.Model):
    nid = models.AutoField(primary_key=True)
    name = models.CharField(max_length=32)
    age = models.IntegerField()

    authorDetail = models.OneToOneField(to='AuthorDetail', to_field='nid',
                                        on_delete=models.CASCADE)  # 就是foreignkey+unique,只不过不需要我们自己来写参数了,并且orm会自动帮你给这个字段名字拼上一个_id,数据库中字段名称为authorDetail_id

#作者详细模型
class AuthorDetail(models.Model):
    nid = models.AutoField(primary_key=True)
    birthday = models.DateField()
    telephone = models.BigIntegerField()
    addr = models.CharField(max_length=64)

#出版社模型
class Publish(models.Model):
    nid = models.AutoField(primary_key=True)
    name = models.CharField(max_length=32)
    city = models.CharField(max_length=32)
    email = models.EmailField()

#书籍模型
class Book(models.Model):
    nid = models.AutoField(primary_key=True)
    title = models.CharField(max_length=32)
    publishDate = models.DateField()
    price = models.DecimalField(max_digits=5,decimal_places=2) #总长度为5,小数位位2
    publish = models.ForeignKey(to='Publish',to_field='nid',on_delete=models.CASCADE) #foreignkey里面可以加很多的参数,to指向表,to_field指向你关联的字段,不写这个,默认会自动关联主键字段,on_delete级联删除
    authors =models.ManyToManyField(to='Author')#注意不管是一对多还是多对多,写to这个参数的时候,最后后面的值是个字符串,不然你就需要将你要关联的那个表放到这个表的上面

第三张表

 

 自己创建第三张表的

class Book(models.Model):
    title = models.CharField(max_length=32,verbose_name='书名')
        
class Author(models.Model):    
    name = models.CharField(max_length=32,verbose_name='作者姓名')
    
 #自己创建第三张表   
class Author2Book(models.Model):
    author = models.ForeignKey(to='Author')
    book = models.ForeignKey(to='Book')
    
    class Meta:
        unique_together = ('author','book')

通过ManytoManyfField自动创建

class Book(models.Model):
    title = models.CharField(max_length=32,verbose_name='书名')

#通过ORM自带的ManyToManyField自动创建第三张表
class Author(models.Model):
    name = models.CharField(max_length=32,verbose_name="作者姓名")
    books = models.ManyToManyField(to='Book',related_name='authors')

设置ManyToManyField并指向自行创建的第三张表

class Book(models.Model):
    title = models.CharField(max_length=32,verbose_name='书名')

#自己创建第三张表,通过ManyToManyField指定关联
class Author(models.Model):
    name = models.CharField(max_length=32,verbose_name='作者姓名')
    books = models.ManyToManyField(to='Book',through='Author2Book',through_fields=('author','book')) #through_fields接收一个2元组('f1','f2'),其中f1是定义ManyToManyField的模型外键的名(author),f2是关联目标模型(book)的外键名。

class Author2Book(models.Model):
    author = models.ForeignKey(to='Author')
    book = models.ForeignKey(to='Book')
    
    class Meta:
        unique_together = ('author','book')

创建一对一时的一些参数

to #设置关联的表
to_field #设置关联的字段
on_delete #同ForeignKey字段

创建一对多时的一些参数

to #设置关联表
to_field #设置关联的字段
on_delete #同ForeignKey字段

创建多对多时的一些参数

to #设置关联表
related_name #同Foreignkey字段
related_query_name #同Foreignkey字段
through #在使用ManyToManyField字段时,Django将自动生成一张表来管理多对多的关联关系。手动创建第三张表时 through来指定第三张表的表名
through_fields #设置关联的字段
db_table #默认创建第三张表时,数据库中表名称

创建表时的一些元信息设置

db_table #ORM在数据库中默认是 app_类名,可以通过db_table重写类名
index_together #联合索引
unique_together #联合唯一索引
ordering #指定默认按什么字段排序 ,只有设置了属性才可以reverse()

on_delete参数

on_delete  #当删除关联表中的数据时,当前表与其关联的行的行为。
models.CASCADE  #删除关联数据,与之关联也删除
models.DO_NOTHING  #删除关联数据,引发错误IntegrityError
models.PROTECT  #删除关联数据,引发错误ProtectedError
models.SET_NULL  #删除关联数据,与之关联的值设置为null(前提FK字段需要设置为可空)
models.SET_DEFAULT  #删除关联数据,与之关联的值设置为默认值(前提FK字段需要设置默认值)
models.SET#删除关联数据,
#a. 与之关联的值设置为指定值,设置:models.SET(值)
#b. 与之关联的值设置为可执行对象的返回值,设置:models.SET(可执行对象)

添加表记录

一对多

方式1:
   publish_obj=Publish.objects.get(nid=1) #拿到nid为1的出版社对象
   book_obj=Book.objects.create(title="金瓶眉",publishDate="2012-12-12",price=100,publish=publish_obj)
  
方式2:
   book_obj=Book.objects.create(title="金瓶眉",publishDate="2012-12-12",price=100,publish_id=1)

多对多

   方式一:
    book_obj=Book.objects.create(title="追风筝的人",price=200,publishDate="2012-11-12",publish_id=1)
    yuan=Author.objects.filter(name="yuan").first()
    egon=Author.objects.filter(name="alex").first() 
    book_obj.authors.add(yuan,egon)    
 方式二
    book_obj.authors.add(1,2)
    book_obj.authors.add(*[1,2]) #这种方式用的最多,因为一般是给用户来选择,用户选择是多选的,选完给你发送过来的就是一堆的id值

多对多其他常用的API

book_obj.authors.remove() # 将某个特定的对象从被关联对象集合中去除
book_obj.authors.remove(*[1,2]) #将多对多的关系数据删除
book_obj.authors.clear() #清空被关联对象集合
book_obj.authors.set() #先清空后设置

基于对象的跨表查询

一对一查询

正向查询(按字段)

egon=Author.objects.filter(name="egon").first()
print(egon.authorDetail.telephone)

反向查询(按表名 不需要_set,查询的就是一条记录)

authorDet=AuthorDetail.objects.filter(addr="北京")[0]
print(authorDet.author.name)

一对多查询

正向查询(按字段)

# 查询id为1的书籍的出版社所在的城市
book_obj = Book.objects.filter(nid=1).first()
print(book_obj.publish.city)

反向查询(按表名 book_set 加上_set是因为反向查询的时候,你查询出来的可能是多条记录的集合)

publish = Publish.objects.get(name='小钱出版社')
book_list = publish.book_set.all()
for book_obj in book_list:
     print(book_obj.title)

多对多查询

正向查询(按字段)

#查询三国演义所有作者的名字及手机号
book_obj = Book.objects.filter(title='三国演义').first()
authors = book_obj.authors.all()
for author_obj in authors: 
     print(author_obj.name,author_obj.authorDetail.telephone)

反向查询(按表名)

# 查询alex出过的所有书籍的名字
    author_obj=Author.objects.get(name="alex")
    book_list=author_obj.book_set.all()        #与alex作者相关的所有书籍
    for book_obj in book_list:
        print(book_obj.title)

 注意

publish = ForeignKey(Book,related_name='bookList')
#查询人民出版社出过的书
publish=Publish.objects.get(name='人民出版社')
book_list=publish.bookList.all()  与人民出版社关联的所有书籍对象集合

基于双下划线的跨表查询

     Django 还提供了一种直观而高效的方式在查询(lookups)中表示关联关系,它能自动确认 SQL JOIN 联系。要做跨关系查询,就使用两个下划线来链接模型(model)间关联字段的名称,直到最终链接到你想要的model 为止。

 基于双下划线的查询就一句话:正向查询按字段,反向查询按表名小写

一对一查询

# 查询alex的手机号(一对一)
# 正向查询
ret=Author.objects.filter(name="alex").values("authordetail__telephone")
# 反向查询
ret=AuthorDetail.objects.filter(author__name="alex").values("telephone")

一对多查询

# 查询北京出版社出版过的所有书籍的名字与价格(一对多) 
# 正向查询 按字段publish
book_obj = Book.objects.filter(publish__name = "北京出版社").values_list('title','price')
# 反向查询 按表名book
publish_obj = Publish.objects.filter(name='北京出版社').values_list('book__title','book__price')

多对多查询

# 查询alex出过的所有书籍的名字(多对多)
#正向查询 按字段authors
book_obj = Book.objects.filter(authors_name = 'alex' ).values_list('title')
#反向查询按表名book
author_obj = Author.objects.filter(name = 'alex').values_list('book_list','book_price')

related_name

反向查询时,如果定义了related_name ,则用related_name替换 表名

publish = ForeignKey(Book, related_name='bookList')

# 练习: 查询人民出版社出版过的所有书籍的名字与价格(一对多)
# 反向查询 不再按表名:book,而是related_name:bookList
publish_obj = Publish.objects.filter(name='人民出版社').values_list('bookList__title','bookList__price')

聚合查询 分组查询 F查询和Q查询

聚合

aggregate(*args,**kwargs)

# 计算所有图书的平均价格
from django.db.models import Avg
Book.objects.all().aggregate(Avg('price'))
或
Book.objects.aggregate(average_price=Avg('price'))

如果你希望生成不止一个聚合,你可以向aggregate()子句中添加另一个参数。所以,如果你也想知道所有图书价格的最大值和最小值,可以这样查询:

from django.db.models import Avg,Max,Min
Book.object.aggregate(Avg('price'),Max('price'),Min('price')) #cpint('id'),count(1)都是统计个数
#输出{'price__avg': 34.35, 'price__max': Decimal('81.20'), 'price__min': Decimal('12.99')}

分组

单表分组查询

查询每一个部门名称以及对应的员工数

emp:

id  name age   salary    dep
1   alex  12   2000     销售部
2   egon  22   3000     人事部
3   wen   22   5000     人事部

sql语句
select dep,Count(*) from emp group by dep;

ORM:
emp.objects.values('dep').annotate(c=Count('id'))

多表分组查询

查询每一个部门名称以及对应的员工数
emp:

id  name age   salary   dep_id
1   alex  12   2000       1
2   egon  22   3000       2
3   wen   22   5000       2

dep

id   name 
1    销售部
2    人事部

emp-dep:

id  name age   salary   dep_id   id   name 
1   alex  12   2000       1      1    销售部
2   egon  22   3000       2      2    人事部
3   wen   22   5000       2      2    人事部


sql语句
select dep.name,Count(*) from emp left join dep on emp_id = dep.id group by dep.id

ORM
dep.objects.values('id').annotate(c=Count('emp')).values('name','c')
ret = models.Emp.objects.values('dep_id','name').annotate(a=Count(1)) 

annotate()为调用的QuerySet中每一个对象都生成一个独立的统计值(统计方法用聚合函数

总结:跨表分组查询本质就是将关联表join成一张表,再按单表的思路进行分组查询,,既然是join连表,就可以使用咱们的双下划线进行连表了

class Emp(models.Model):
    name = models.CharField(max_length=32)
    age = models.IntegerField(max_length=32)
    salary = models.DecimalField(max_digits=8,decimal_places=2)
    dep = models.CharField(max_length=32)
    province = models.CharField(max_length=32)

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

# 单表
#查询每一个部门的id以及对应员工的平均薪水
ret = models.Emp.objects.values('dep_id').annotate(s = Avg('salary'))
#查询每个部门的id以及对对应的员工的最大年龄
ret = models.Emp.objects.values('dep_id').annotate(a = Max('age'))
#连表
# 查询每个部门的名称以及对应的员工个数和员工最大年龄
ret = models.Emp.objects.values('dep_name').annotate(a = Count('id'),b =Max('age'))

annotate的返回值是querySet,如果不想遍历对象,可以用上value_list

练习题

#聚合查询
from django.db.models import Avg,Min,Max,F,Q,Count,Sum
ret  = models.Book.objects.all().aggregate(s=Avg('price'))
ret = models.Book.ojects.aggregate(s=Avg('price'))
ret ={'s':平均值}

#分组查询
annotate(c=Count('id'),m=Max(),min=Min())

# 查询每个作者的姓名以及出版的书的最高价格
models.Author.objects.values('id','name').annotate(m=Max('book_price'))
models.Author.objects.annotate(m=Max('book_price')).values('name','m')
# values写在前面,直接就是筛选出结果字段,这些字段同时作为分组依据
# values写在后面,是通过调用者表的id进行分组,得到结果的结果是queryset(model对象),再通过values取值

# 查询作者id大于2作者的姓名以及出版的书的最高价格
models.Author.objects.filter(id__gt=2).values('id','name').annotate(m=Max('book_price'))
models.Author.objects.filter(id__gt=2).annotate(m=Max('book_price')).values('name','m')

# 查询作者(id大于2或者作者年龄大于等于20岁)的女作者的姓名以及出版的书的最高价格
models.Author.objects.filter(Q(id__gt=2)|Q(age__gt=20),sex='').values('id','name').annotate(m=Max('book_price'))
models.Author.objects.filter(Q(id__gt=2)|Q(age__gt=20),sex='').annotate(m=Max('book_price')).values('m','name')

# 查询每个作者出版的书的最高价格 的平均值
models.Author.objects.values('id','name').annotate(m=Max("book_price")).aggregate(a=Avg('m'))

F查询与Q查询

F查询

# 我们在book表里面加上两个字段:评论数:commentNum,收藏数:KeepNum
# 查询评论数大于收藏数的书籍
from django.db.models import F,Q
Book.objects.filter(commentNum__lt=F('keepNum'))
# 查询评论数大于收藏数2倍的书籍
Book.objects.filter(commentNum__lt=F('keepNum')*2)
# 修改操作也可以使用F函数,比如将每一本书的价格提高30元
Book.objects.all().update(price = F('price')+30)

Q查询

filter() 等方法中的关键字参数查询都是一起进行“and” 的

from django.db.models import Q
Q(title_startswitch='Py')
# Q 对象可以使用&(与) 、|(或)、~(非) 操作符组合起来。
book_obj = Book.objects.filter(Q(authors__name='alex')|Q(authors__name='wusir'))
# Q对象可以使用~取反
book_obj = Book.objects.filter(Q(authors__name='alex')&~Q(authors__name='wusir'))
# 如果出现Q 对象,它必须位于所有关键字参数的前面
bo

ORM执行原生sql语句

raw()管理器方法用于原始的SQL查询,并返回模型的实例

注意:raw()语法查询必须包含主键

ret = models.Student.objects.raw('select id, tname as hehe from app02_teacher')
    for i in ret:
        print(i.id, i.hehe)

raw()方法自动将查询字段映射到模型字段。还可以通过translations参数指定一个把查询的字段名和ORM对象实例的字段名互相对应的字典

d = {'tname': 'haha'}
    ret = models.Student.objects.raw('select * from app02_teacher', translations=d)
    for i in ret:
        print(i.id, i.sname, i.haha)

原生SQL还可以使用参数,注意不要自己使用字符串格式化拼接SQL语句,防止SQL注入!

执行自定义SQL

from django.db import connection,connections

cursor = connection.cursor()
cursor.execute('''select * from auth_user where id = %s''',params = [1,])
ret = cursor.fetchone()

python脚本调用Django环境

import os
if __name__ == '__main__':
    os.environ.setdefault("DJANGO_SETTINGS_MODULE", "moxing.settings")
    import django
    django.setup()
    
    from app01 import models
    books = models.Book.objects.all()
    print(books)

 

参考https://www.cnblogs.com/clschao/articles/10439958.html

posted @ 2019-03-04 16:56  等待の喵  阅读(497)  评论(0编辑  收藏  举报