zoukankan      html  css  js  c++  java
  • Django框架(十)—— 多表操作:一对一、一对多、多对多的增删改,基于对象/双下划线的跨表查询、聚合查询、分组查询、F查询与Q查询

    多表操作:增删改,基于对象/双下划线的跨表查询、聚合查询、分组查询、F查询与Q查询

    一、创建多表模型

    一对一:OneToOneField

    一对多:ForeignKey

    多对多:ManyToManyField

    • 创建表时,会自动添加一个nid字段,并且自增,所以id可以不用手动创建

    • OneToOneField和ForeignKey会自动在后面加上" _id "

    • ManyToManyField会自动创建第三张表

    • 创建表的类,OneToOneField、ForeignKey和ManyToManyField中的to后的主表,
      1、如果用双引号包裹,那么创建主表的类在上在下都可以

      2、如果不用双引号包裹,那么创建主表的类就必须在从表的上方

    二、一对多增删改表记录

    class Publish(models.Model):
        # id如果不写,会自动生成,名字叫nid,并且自增
        id = models.AutoField(primary_key=True)
        name = models.CharField(max_length=32)
        addr = models.CharField(max_length=64)
        email = models.EmailField()
    
    
    class Author(models.Model):
        id = models.AutoField(primary_key=True)
        name = models.CharField(max_length=32)
        # 数字类型
        sex = models.IntegerField()
        # to='AuthorDetail'  加引号,这个表能找到就可以,不用引号,类必须在上面定义
        authordetail = models.OneToOneField(to='AuthorDetail', to_field='id')
    
        def __str__(self):
            return self.name
    
    
    class AuthorDetail(models.Model):
        id = models.AutoField(primary_key=True)
        phone = models.CharField(max_length=32)
        addr = models.CharField(max_length=64)
    
    
    class Book(models.Model):
        id = models.AutoField(primary_key=True)
        name = models.CharField(max_length=32)
        price = models.DecimalField(max_digits=5, decimal_places=2)
        publish = models.ForeignKey(to=Publish, to_field='id')
    
        authors = models.ManyToManyField(to=Author)
    
        def __str__(self):
            return self.name

    1、一对多添加记录

    (1)通过 _id 来添加记录

    models.Book.objects.create(name='红楼梦',price=20.0,publish_id=1)

    (2)通过对象添加记录

    # pk 表示主键,一旦主键改变,还是会找到对应的主键来获取对象
    publish = models.Publish.objects.filter(pk=1).first()
    models.Book.objects.create(name='红楼梦',price=34.5,publish=publish)

    2、一对多删除记录

    删除记录同单表删除记录一样

    models.Book.objects.filter(name='红楼梦').first().delete()

    3、一对多修改记录

    (1)通过queryset对象的update方法修改

    # 方式一:通过 _id 修改
    models.Book.objects.filter(pk=1).update(name='西游记',publish_id=1)
    
    # 方式二:通过对象修改
    publish = models.Publish.objects.filter(pk=1).first()
    models.Book.objects.filter(pk=1).update(name='西游记',publish=publish)

    (2)通过对象的属性修改,调用对象的save()方法保存

    # 方式一:通过 _id 修改
    book = models.Book.objects.filter(pk=1).first()
    book.publish_id = 2
    book.save()
    
    # 方式二:通过对象修改
    publish = models.Publish.objects.filter(pk=2).first()
    book.publish = publish
    book.save()

    三、一对一增删改记录

    一对一的增删改记录和一对多的方法相同,参考一对多的增删改记录

    四、多对多增删改记录

    1、多对多增加记录

    add()    # 可以传对象,可以传id,可以传多个

    (1)通过对象添加记录

    # 为红楼梦这本书添加一个作者
    tom = models.Author.objects.filter(name='tom').first()
    book = models.Book.objects.filter(name='红楼梦').first()
    tony = models.Author.objects.filter(name='tony').first()
    jack = models.Author.objects.filter(name='jack').first()
    book.authors.add(tom)
    # 一次添加两个作者,中间用逗号隔开
    book.authors.add(tony,jack)

    (2)通过 id 添加记录

    book = models.Book.objects.filter(name='红楼梦').first()
    book.authors.add(1)
    # 一次添加两个作者,中间用逗号隔开
    book.authors.add(1,2)

    2、多对多删除记录

    remove()    # 可以传对象,可以传id,可以传多个

    (1)通过对象删除记录

    # 删除红楼梦作者名字是tom的作者
    tom = models.Author.objects.filter(name='tom').first()
    book = models.Book.objects.filter(name='红楼梦').first()
    book.authors.remove(tom)

    (2)通过id删除记录

    # 删除红楼梦作者id是1、2的两个作者
    book = models.Book.objects.filter(name='红楼梦').first()
    book.authors.remove(1,2)

    3、多对多清空记录

    clear()    # 没有参数
    # 清除红楼梦的所有作者
    book = models.Book.objects.filter(name='红楼梦').first()
    book.authors.clear()

    4、多对多修改记录

    set()    # 先清空在增加,必须传列表,列表里面可以是对象,可以是id

    (1)通过对象修改记录

    # 修改红楼梦作者名字为tom
    tom = models.Author.objects.filter(name='tom').first()
    book = models.Book.objects.filter(name='红楼梦').first()
    book.authors.set([tom,])

    (2)通过id修改记录

    # 修改红楼梦作者为作者id是1、2的作者
    book = models.Book.objects.filter(name='红楼梦').first()
    book.authors.set([1,2])

    五、基于对象的跨表查询--多次查询、子查询

    正向查询和反向查询

    正向查询:关联字段在从表,由从表查询主表中的数据    -----> 按字段查询
    反向查询:关联字段在从表,由主表查询从表中的数据    -----> 按表名小写查询

    1、一对一基于对象的跨表查询

    正向:正向查询按 字段
    反向:反向查询按 表名小写
    # 1.查询tom作者的手机号   正向查询
    author=Author.objects.filter(name='tom').first()
        # author.authordetail 就是作者详情的对象
    authordetail=author.authordetail
    print(authordetail.phone)
    
    # 2. 查询地址是山东的作者名字   反向查询
    authordetail=AuthorDetail.objects.filter(addr='山东').first()
        # authordetail.author  这是作者对象
    author=authordetail.author
    print(author.name)

    2、一对多基于对象的跨表查询

    正向:正向查询按 字段
    反向:反向按 表名小写_set.all()
    # 1.正向 查询红楼梦这本书的出版社邮箱
    book=Book.objects.filter(name='红楼梦').first()
        # book.publish  就是出版社对象
    pulish=book.publish
    print(pulish.email)
    
    # 2.反向  查询地址是北京的出版社出版的图书
    publish=Publish.objects.filter(addr='北京').first()
        # publish.book_set.all()  拿出所有的图书
    books=publish.book_set.all()
        # 统计一下条数
    books=publish.book_set.all().count()
    print(books)

    3、多对多基于对象的跨表查询

    正向:正向查询按 字段.all()
    反向:反向按 表名小写_set.all()
    # 1.查询红楼梦这本书所有的作者
    book=Book.objects.filter(name='红楼梦').first()
    book.authors.all()  # 是所有的作者,是一个queryset对象,可以继续点
    print(book.authors.all())
    
    # 2.查询lqz写的所有书
    lqz=Author.objects.filter(name='lqz').first()
    books=lqz.book_set.all()
    print(books)

    六、基于双下划线的跨表查询

    1、一对一的基于双下划线的跨表查询

    正向:按 字段,跨表可以在filter,也可以在values中
    反向:按 表名小写,跨表可以在filter,也可以在values中
    # 1.查询tom作者的手机号   正向查询  跨表的话,按字段
       # 以author表作为基表
        ret=Author.objects.filter(name='tom').values('authordetail__phone')
        print(ret)
        
    # 2.以authordetail作为基表 反向查询,按表名小写  跨表的话,用表名小写
        ret=AuthorDetail.objects.filter(author__name='tom').values('phone')
        print(ret)

    2、一对多的基于双下划线的跨表查询

    正向:按 字段,跨表可以在filter,也可以在values中
    反向:按 表名小写,跨表可以在filter,也可以在values中
    # 查询出版社为北京出版社出版的所有图书的名字,价格
    # 1.以Publish为基表
    ret=Publish.objects.filter(name='北京出版社').values('book__name','book__price')
    print(ret)
    
    # 2.以Book为基表
    ret=Book.objects.filter(publish__name='北京出版社').values('name','price')
    print(ret)

    3、多对多的基于双下划线的跨表查询

    正向:按 字段,跨表可以在filter,也可以在values中
    反向:按 表名小写,跨表可以在filter,也可以在values中
    # 查询红楼梦的所有作者名字
    # 1.以Book为基表
    ret=Book.objects.filter(name='红楼梦').values('authors__name')
    print(ret)
    
    # 2.以Author为基表
    ret = Author.objects.filter(book__name='红楼梦').values('name')
    print(ret)

    4、连续跨表查询

    一直用双下划线 __ 获取字段

    # 查询红楼梦这本书所有的作者的手机号
    book=Book.objects.filter(name='红楼梦').first()
    authors=book.authors.all()
    for author in authors:
        authordetail=author.authordetail
        print(authordetail.phone)
    
    # 方法二,使用连续跨表
    ret = Book.objects.filter(name='红楼梦').values('authors__authordetails__phone')
    print(ret)

    七、聚合查询

    1、什么聚合

    用聚合函数来汇总多条信息

    2、语法

    from django.db.models import Avg,Sum,Count,Max,Min
    aggregate(*args, **kwargs)

    3、聚合函数使用

    # 聚合函数
    Max('price'),Min('price'),Avg('price'),Sum('price')
    # 计算所有图书的平均价格
    ret = models.Book.objects.all().aggregate(Avg('price'))
    print(ret)
    
    # 计算图书的最高价格,最低价格,平均价格,总价
    ret = models.Book.objects.all().aggregate(Max('price'),Min('price'),Avg('price'),Sum('price'))
    print(ret)

    八、分组查询

    1、关键注意点

    values在前,表示group by,在后,表示取值
    filter在前,表示过滤(where),在后,表示having(对分组之后的结果再进行过滤)
    

    2、分组查询案例

    # 查询所有作者写的书的总价格大于26的
    # filter()在annotate后面,表示对分组后的结果进行筛选,相当于having
    # annotate前的values()表示按该字段分组,相当于group by,可以省略,默认会按Author的id分组
    # 后面的values()表示取值
    ret=Author.objects.all().values('pk').annotate(s=Sum('book__price')).filter(s__gt=26).values('name','s')
    
    等价于
    
    ret=Author.objects.all().annotate(s=Sum('book__price')).filter(s__gt=26).values('name','s')
    # 查询各个作者出的书的总价格
    # s相当于给求和结果取名字,在vlaues取值中,可以引用
    ret = Author.objects.all().annotate(s=Sum('price')).values('name','s')
    print(ret)
    # 统计不止一个作者的图书
    ret=Book.objects.all().values('pk').annotate(c=Count('authors')).filter(c__gt=1).values('name','c')
    print(ret)
    
    等价于
    
    ret = Book.objects.annotate(author_num=Count("authors")).filter(author_num__gt=1).values('name','author_num')
    print(ret)

    九、F查询和Q查询

    1、F查询

    (1)什么是F查询

    如果要对两个字段的值作比较,就不能直接比较,必须要借助F() 的实例,它可以在查询中引用字段,来比较同一个 model 实例中两个不同字段的值。

    (2)语法

    from django.db.models import F
    fliter(commit_num__gt=F('read_num'))

    (3)F查询的使用

    # 查询评论数大于阅读数的书
    book = models.Book.objects.filter(commit_num__gt=F('read_num')).values('name')
    
    # 把所有书的评论数加1
    models.Book.objects.all().update(commit_num=F('commit_num')+1)

    2、Q查询

    (1)什么是Q查询

    对两个字段的值进行 且、或、非 运算,不能直接运算,需要用Q()的实例

    (2)语法

    # 且  ----->  &
    # 或  ----->  |
    # 非  ----->  ~
    from django.db.models import Q
    fliter(Q(name='tom')|Q(name='tony'))

    (3)Q查询的使用

    # 查询不是tom写的书的名字
    ret = models.Book.objects.filter(~Q(authors__name='tom')).values('name')
    
    # 查询作者名字是tom或者名字是tony的书
    ret = models.Book.objects.filter(Q(authors__name='tom')|Q(authors__name='tony')).values('name')
    博客内容仅供参考,部分参考他人优秀博文,仅供学习使用
  • 相关阅读:
    spring容器与java访问权限的关系!
    Mybatis初始化过程详解
    Spring-boot 一些常用注解说明!
    JNDI + Spring 如何读取数据
    HashMap,,ConcurrentHashMap------------------浅谈!!
    spring ioc容器和spring mvc 容器--------浅谈!!
    针对xml文件做提取与写入的操作
    vim编辑器显示行号
    ubuntu双系统修复windows引导
    git基本操作
  • 原文地址:https://www.cnblogs.com/zhuzhiwei-2019/p/10779132.html
Copyright © 2011-2022 走看看