继续上面一篇文章的内容,本文介绍多表操作。使用django ORM可以创建多表关系,并且也支持多张表之间的操作,以创建表关系和查询两部分说明django ORM的多表操作。以作者、图书、出版社和作者信息几张表作为案例进行说明。
创建表关系
注意:在实际开发中不推荐使用外键建立表关系即不使用级联更新和级联删除,而是推荐使用逻辑上的外键关系建立表关系。
上述的四张表中,图书和出版社这两表的关系属于一对多的关系,外键建立在查询频率高的一方。作者和作者详情表属于一对一关系,外键建立在查询频率高的一方,作者和图书属于多对多关系,需要第三张表存储关系,建议将外键建在查询频率高的一方。创建表时一定要执行数据库迁移命令哦~
创建表关系时可以先将表模型创建出来,然后再添加外键字段,另外在使用django ORM创建外键关系时,关联的外键字段会自动在字段后加
_id
,表与表之间的关系默认以主键作为关联字段。另外在创建表关系时,不建议使用实质的外键进行关联,而是通过使用逻辑上的关系来指定表关系。
class Book(models.Model):
name = models.CharField(max_length=60, verbose_name='图书名')
price = models.DecimalField(max_digits=8, decimal_places=2, verbose_name='图书价格')
inventory_num = models.IntegerField(verbose_name='库存数量')
sell_num = models.IntegerField(verbose_name='卖出数量')
publish = models.ForeignKey(to='Publish', on_delete=models.DO_NOTHING, db_constraint=False, verbose_name='外键关联出版社')
author = models.ManyToManyField(to='Author', on_delete=models.DO_NOTHING, db_constraint=False, verbose_name='外键关联作者')
class Publish(models.Model):
name = models.CharField(max_length=12, verbose_name='出版社名称')
class Author(models.Model):
name = models.CharField(max_length=10, verbose_name='作者名称')
author_detail = models.OneToOneField(to='AuthorDetail', on_delete=models.DO_NOTHING, db_constraint=False, verbose_name='外间关联作者详情')
class AuthorDetail(models.Model):
age = models.IntegerField(verbose_name='年龄')
phone = models.CharField(max_length=11, verbose_name='手机号')
另外还需补充一点,多对多的表关系共有三种创建方式,分别是全自动创建、半自动创建和全手动创建:
使用全自动创建多对多关系的优点就是无需手动创建第三张表,非常方便,django ORM直接提供操作第三张表关系的方法
缺点就是无法扩展第三张关系表
class Book(models.Model):
name = models.CharField(max_length=32)
authors = models.ManyToManyField(to='Author')
class Author(models.Model):
name = models.CharField(max_length=32)
第三张表完全取决于手动的扩展,但是需要写的代码较多,而且无法使用ORM提供的简单方法
class Book(models.Model):
name = models.CharField(max_length=32)
class Author(models.Model):
name = models.CharField(max_length=32)
class Book2Author(models.Model):
book_id = models.ForeignKey(to='Book')
author_id = models.ForeignKey(to='Author')
class Book(models.Model):
name = models.CharField(max_length=32)
authors = models.ManyToManyField(
to='Author',
through='Book2Author',
through_fields=('book','author')
class Author(models.Model):
name = models.CharField(max_length=32)
class Book2Author(models.Model):
book = models.ForeignKey(to='Book')
author = models.ForeignKey(to='Author')
多表数据操作 - 增删改
首先介绍多表操作的增删改操作,因为多表的查询数据操作稍微麻烦一点,单独另外开小灶。
一对多&一对一关系 - 增删改
一对一和一对多的增删改操作基本是一致的。
增加数据有两种方式,一种方式是通过实际字段来添加,另一种方式是通过虚拟字段对象赋值添加。
book_obj = models.Book.objects.create(name='哈利波特', price=10.2, publish_id=1)
publis_obj = models.Publish.objects.filter(pk=1).first()
book_obj = models.Book.objects.create(name='哈利波特', price=10.2, publish=publis_obj)
需要说明一点,在实际项目开发中删除数据并不是真的删除了,而是使用一个布尔类型的字段标识该数据是否删除。
删除数据的时候如果不指定on_delete=models.DO_NOTHING
默认是级联更新级联删除的。
models.Publish.objects.filter(pk=1).delete()
修改数据同增加数据一样有两种方式。
models.Book.objects.filter(pk=1).update(publish_id=1)
pub_obj = models.Publish.objects.filter(pk=2).first()
models.Book.objects.filter(pk=1).update(publish=pub_obj)
多对多关系 - 增删改
首先需要明确的是,多对多的增删改是在操作第三张关系表,但是第三张关系表是django自动创建的,如何通过代码进入第三张表呢?多对多的外键关系被建在book表,多对多的外键字段是author
字段,因此通过book_obj.author
即可操作第三张表了。
对于django自动创建的第三张表的多对多关系,django提供了额外的方法对数据进行操作。
增加多对多关系 - add()
add()
方法给第三张关系表添加数据,括号内既可以传数字也可以传对象,并且都支持多个同时操作。
book_obj = models.Book.objects.filter(pk=1).first()
book_obj.author.add(1)
book_obj.author.add(2, 3)
book_obj = models.Book.objects.filter(pk=2).first()
author_obj1 = models.Author.objects.filter(pk=1).first()
author_obj2 = models.Author.objects.filter(pk=2).first()
author_obj3 = models.Author.objects.filter(pk=3).first()
book_obj.author.add(author_obj1)
book_obj.author.add(author_obj2, author_obj3)
删除多对多关系 - remove()
remove()
方法用来为第三张表删除数据,同样的,括号内既可以传数字也可以传对象,并且支持多条数据同时操作。
book_obj = models.Book.objects.filter(pk=1).first()
book_obj.author.remove(2)
book_obj.authors.remove(1, 3)
author_obj1 = models.Author.objects.filter(pk=2).first()
author_obj2 = models.Author.objects.filter(pk=3).first()
book_obj.authors.remove(author_obj1, author_obj2)
修改多对多关系 - set()
set()
方法用来修改第三张表,该方法是一个覆盖操作,用新的关系覆盖之前的关系,该方法的参数必须是一个列表或者元组,指出数字或对象,也支持多条数据同时操作。
book_obj = models.Book.objects.filter(pk=1).first()
book_obj.author.set([2])
book_obj.authors.set([1, 2])
author_obj2 = models.Author.objects.filter(pk=2).first()
author_obj3 = models.Author.objects.filter(pk=3).first()
book_obj.authors.set([author_obj2, author_obj3])
清空第三张表某个对象的绑定关系 - clear()
clear()
方法会清空第三张关系表中某个对象的绑定关系。
book_obj = models.Book.objects.filter(pk=1).first()
book_obj.author.clear()
在进行多表查询操作前,需要了解一个概念,什么是正向查询和反向查询。
正向查询:外键在哪个表中,查询关联的表就是正向查询,比如通过书籍查询出版社;
反向查询:被关联的表查外键字段所在的表就是反向查询,比如通过出版社查询书籍。
如果查询比较复杂时可以采用子查询的方式,子查询就是分步骤查询的意思,先查询得到的结果作为后查询的条件。
正向查询按字段,如果有多个结果需要外键字段.all()
,那么怎么判断查询的结果有多个呢?如果在不加.all()
的情况下得到的结果是应用名.模型名.None
比如first.Author.None
这种情况下说明ORM 语句没有错误,只是查询到的结果有多个,就需要加.all()
。当查询结果只有一个时得到的是一个模型对象,如果为多个就是QuerySet对象。
book_obj = models.Book.objects.filter(pk=1).first()
res = book_obj.publish
print(res.name)
book_obj = models.Book.objects.filter(pk=1).first()
res = book_obj.author
res_many = book_obj.author.all()
author_obj = models.Author.objects.filter(name='lili').first()
res = author_obj.author_detail
print(res.phone)
反向查询如果是一对一的话是表名小写
,如果是一对多或者多对多时反向查询是表明小写——set
,另外如果有多个结果需要在表明小写后再加_set.all()
,判断结果是否有多个的方法与正向查询相同。当查询结果只有一个时得到的是一个模型对象,如果为多个就是QuerySet对象。
publish_obj = models.Publish.objects.filter(name='东方').first()
res = publish_obj.book_set.all()
author_obj = models.Author.objects.filter(name='lili').first()
res = author_obj.book_set
res_many = author_obj.book_set.all()
print(res_many)
author_detail_obj = models.AuthorDetail.objects.filter(phone='119').first()
res = author_detail_obj.author
print(res.name)
联表查询就是像MySQ里面SQL语句的联表查询一样,只不过在django的ORM里面使用基于双下划线联表查询(跨表查询)。联表查询的可以使用一行代码查询结果,联表查询也遵循正反向关系。
res = models.Book.objects.filter(pk=1).values('name', 'publish__name')
res = models.Book.objects.filter(pk=1).values('author__name')
res = models.Author.objects.filter(name='lili').values('name', 'author_detail__phone').first()
print(res.get('name'), res.get('author_detail__phone'))
res = models.Publish.objects.filter(book__id=1).values('name', 'book__name')
res = models.Author.objects.filter(book__id=1).values('name', 'book__name')
res = models.AuthorDetail.objects.filter(author__id=1).values('author__name', 'phone')
res = models.Book.objects.filter(pk=1).values('author__author_detail__phone')
聚合查询通常情况下是配合分组一起使用的,聚合查询就是用一些统计工具,比如最大值,最小值,平均值等,聚合函数的导入方式from django.db.models import Max, Min, Sum, Count, Avg
,如果在不分组的情况下使用聚合函数需要在aggregate()
方法内使用。
from django.db.models import Min,Max,Sum,Count,Avg
res = models.Book.objects.aggregate(Avg('price'))
print(res)
res = models.Book.objects.aggregate(Max('price'),Sum('price'),Count('pk'))
print(res)
聚合函数通常和分组一起使用,分组查询的方法是annotate
,默认以models.分组依据
作为分组依据,即表的主键进行分组,如果annotate()
方法前面出现了values()
那么就会按照values
中指定的值进行分组。分组查询支持__
跨表查询。
from django.db.models import Sum, Max, Min, Avg, Count
res = models.Book.objects.annotate(author_num=Count('author')).values('name', 'author_num')
res = models.Publish.objects.annotate(min_price=Min('book__price')).values('name', 'min_price')
res = models.Book.objects.annotate(author_num=Count('author')).filter(author_num__gt=1).values('name', 'author_num')
res = models.Author.objects.annotate(sum_price=Sum('book__price')).values('name', 'sum_price')
F与Q查询
F查询可以获得表中某个字段的数据值,尤其适合表中两个字段之间的比较运算,在操作字符类型的数据时,F不能直接做字符串的拼接,需要借助Concat和Value
。
from django.db.models import F
res = models.Book.objects.filter(sell_num__gt=F('inventory_num'))
res = models.Book.objects.update(price=F('price')+20)
F查询对于字符串的操作需要借助Concat
和Value
两个方法:
from django.db.models.functions import Concat
from django.db.models import F, Value
models.Book.objects.update(name=Concat(F('name'),Value('爆款')))
使用filter()
进行条件过滤时,采用的是逻辑与and
的操作,如果想要将多个筛选条件更改为or
或者not
的关系则需要借助Q查询。在Q查询中|
表示or
的关系,~
表示not
的关系。
import django.db.models import Q
res = models.Book.objects.filter(~Q(sell_num__gt=100) | Q(price__lt=20))
另外Q查询还有另一个比较高级的用法,就是可以将查询条件的左边也变成字符串的形式。
q = Q()
q.connector = 'or'
q.children.append(('sell_num__gt',100))
q.children.append(('price__lt',200))
res = models.Book.objects.filter(q)
print(res)
django开启事务
MySQL为了保证数据的安全有一个事务的机制,django既然能够连接MySQL那么django就可以支持MySQL的事务机制
。下述代码就是在django中开启事务:
from django.db import transaction
try:
with transaction.atomic():
except Exception as e:
print(r)
文章首发于微信公众号程序媛小庄,同步于掘金。
码字不易,转载请说明出处,走过路过的小伙伴们伸出可爱的小指头点个赞再走吧(╹▽╹)