关联对象操作及多表查询

关联表的数据操作:

一对多:

正向:如果一个模型有外键字段,通过这个模型对外键进行操作叫做正向。

更新:

  通过属性赋值

In [1]: from teacher.models import Student                                                                                                          

In [2]: from teacher.models import Grade                                                                                                            

In [3]: g=Grade.objects.create(name='django框架',num='7')                                                                                           

In [4]: g
Out[4]: <Grade: Grade object (1)> In [5]: g=Grade.objects.first() In [6]: s=Student.objects.first() In [7]: s.grade=g In [8]: s.save()

  通过主键的方式

In [9]: s2=Student.objects.last()                                       

In [10]: s2.grade_id=g.id                                               

In [11]: s2.save()

  总结:Foreignkey字段的更新,跟普通字段没什么区别。

删:只有外键设置了null=True,你就可以通过赋值None来删除关系

In [15]: s.grade=None                                                   

In [16]: s.save()

查:

In [18]: res=Student.objects.filter(grade__name='django框架')           

In [19]: print(res.query)
SELECT `teacher_student`.`id`, `teacher_student`.`name`, `teacher_student`.`age`, `teacher_student`.`sex`, `teacher_student`.`qq`, `teacher_student`.`phone`, `teacher_student`.`c_time`, `teacher_student`.`grade_id` FROM `teacher_student` INNER JOIN `teacher_grade` ON (`teacher_student`.`grade_id` = `teacher_grade`.`id`) WHERE `teacher_grade`.`name` = django框架

过滤Foreignkey字段模型的字段

 反向:一个模型如果被另外一个模型外键关联,通过这个模型对关联它的模型进行操作叫做反向。

  查:通过管理器,默认管理器是有外键的模型名称的小写加上_set(foo_set,foo是模型名称小写),通过这个管理器可以查询模型的实例,在定义外键的时候,通过related_name可以覆盖这个名称。

grade=models.ForeignKey('Grade',on_delete=models.SET_NULL,null=True,related_name='students1')

In [2]: from teacher.models import Grade,Student                                                            

In [3]: g=Grade.objects.first()                                                                             

In [4]: g
Out[4]: <Grade: django框架> In [6]: g.students1.all()
Out[6]: <QuerySet [<Student: 赵柳>]>

  增:

    通过add方法,可以添加多个

In [11]: new_s=Student.objects.first()   
In [13]: g.students1.add(new_s)   
In [17]: s1=Student.objects.get(pk=2)
In [18]: s2=Student.objects.get(pk=3)
In [19]: g.students1.add(s1,s2)

  

    通过create方法

In [22]: g.students1.create(name='郭富城')
Out[22]: <Student: 郭富城>

  删:删掉关系

    remove(obj1,obj2,obj3)

In [23]: gfc=Student.objects.last()                                                                         

In [24]: g.students1.remove(gfc)

    clear() 清空

In [25]: g.students1.clear()

  

  add,remove,clear直接操作数据库

  改:替换对象集

  set([s1,s2])

In [25]: s1
Out[25]: <Student: 李四> In [26]: s2
Out[26]: <Student: 王五> In [27]: g.students1.set([s1,s2])

多对多:

如果因为有额外字段,自定义了中间模型,我们就需要通过中间模型的管理器进行manytomany关系的创建和删除

默认情况跟一对多中的add,create,remove,clear用法一致

唯一的区别是多对多正向的时候,多对多字段就是一个管理器。

反向的时候,跟一对多的方向一致,也是在模型小写后面加上_set,和一对多类似,一样可以通过定义related_name可以覆盖这个名称。

In [1]: from teacher.models import Course,Enroll                                                                                                    

In [2]: c1=Course.objects.create(name='python全栈')   

In [4]: c2=Course.objects.create(name='python全套')   

In [10]: from teacher.models import Course,Enroll,Student                                                                                           

In [11]: s1=Student.objects.create(name='范冰冰')                                                                                                   

In [12]: s2=Student.objects.create(name='李冰冰')                                                                                                   

In [13]: s1
Out[13]: <Student: 范冰冰> In [14]: s2
Out[14]: <Student: 李冰冰> In [15]: e=Enroll() In [16]: e.student=s1 In [17]: e.course=c1 In [18]: e.paid=6680 In [19]: e.save() In [20]: c1
Out[20]: <Course: python全栈> In [21]: c1.students.all()
Out[21]: <QuerySet [<Student: 范冰冰>]> In [22]: c1.students.filter(name='范冰冰')
Out[22]: <QuerySet [<Student: 范冰冰>]> In [23]: c1.students.get(name='范冰冰')
、Out[23]: <Student: 范冰冰> In [24]: s1
Out[24]: <Student: 范冰冰> In [25]: s1.course_set.filter(name__startswith='python')
Out[25]: <QuerySet [<Course: python全栈>]>

一对一:

非常类似一对多字段,增删改查和普通字段没有什么区别

反向的时候,使用模型的小写,也可以给related_name覆盖,这个就不是管理器,就是一个普通属性。

注意:一个被一对一管理的模型,他的实例如果没有被分配关系。举个例子:学生对象,没有分配一个学生详情对象。如果去取,全抛出异常DoseNotExist。

跨表查询: 要跨越关系,只需使用跨越模型的相关字段名,以下划线分隔,直到达到你想要的字段为止。 

例如:查询男生都报名了什么课程

res=Course.objects.filter(students__sex=1).distinct()

  这个关系要多深就有多深

例:查询所有报名了python课的学员

res=Student.objects.filter(course__name__contains='python')

例:查询所有报名python全栈课程,在django框架第七期班级的学员

res=Student.objects.filter(course__name='python全栈',grade__name='django框架',grade__num='7')

例:学员报名了python课程的班级有哪些

res=Grade.objects.filter(students1__course__name__contains='python').distinct()

  

Django模型三的更多相关文章

  1. {django模型层(二)多表操作}一 创建模型 二 添加表记录 三 基于对象的跨表查询 四 基于双下划线的跨表查询 五 聚合查询、分组查询、F查询和Q查询

    Django基础五之django模型层(二)多表操作 本节目录 一 创建模型 二 添加表记录 三 基于对象的跨表查询 四 基于双下划线的跨表查询 五 聚合查询.分组查询.F查询和Q查询 六 xxx 七 ...

  2. django模型

    用django时,只要用到数据库就得用到模型. 一.数据库的MTV开发模式 从MVC到MTV 所谓软件架构的MVC模式将数据的存取逻辑(Module),表现逻辑(View)和业务逻辑(Controll ...

  3. 【Python】django模型models的外键关联使用

    Python 2.7.10,django 1.8.6 外键关联:http://www.bubuko.com/infodetail-618303.html 字段属性:http://www.cnblogs ...

  4. django 模型层(2)

    Django 模型层(2) 多表操作---模型之间的关系 1 一对一:作者----作者详细信息 2 一对多:书籍----出版社 3 多对多:书籍----作者 一  创建模型(主键(id)自动创建) 没 ...

  5. django模型系统(二)

    django模型系统(二) 常用查询 每一个django模型类,都有一个默认的管理器,objects QuerySet表示数据库中对象的列表.他可以有0到国歌过滤器.过滤器通过给定参数,缩小查询范围( ...

  6. Django(三)框架之第二篇

    https://www.cnblogs.com/haiyan123/p/7717788.html 一.知识点回顾 1.MTV模型 model:模型,和数据库相关的 template:模板,存放html ...

  7. Django 学习第六天——Django模型基础第一节

    一.Django 的 ORM 简介: Django的ORM系统的分析: 1.ORM 概念:对象关系映射(Object Relational Mapping,简称ORM) 2.ORM的优势:不用直接编写 ...

  8. Django模型层(2)

    <!DOCTYPE html><html lang="zh-cn"><head><meta charset="utf-8&quo ...

  9. django模型中的关系对应

    显然,关系数据库的力量在于将表相互关联.Django提供了定义三种最常见的数据库关系类型的方法:多对一,多对多和一对一. 在说明之前,首先来理解一下这三个概念: 多对一: 两个集合a,b;集合a中的多 ...

随机推荐

  1. HDU-1034 Candy Sharing Game 模拟问题(水题)

    题目链接:https://cn.vjudge.net/problem/HDU-1034 水题 代码 #include <cstdio> #include <algorithm> ...

  2. BZOJ 2141 排队(分块+树状数组)

    题意 第一行为一个正整数n,表示小朋友的数量:第二行包含n个由空格分隔的正整数h1,h2,…,hn,依次表示初始队列中小朋友的身高:第三行为一个正整数m,表示交换操作的次数:以下m行每行包含两个正整数 ...

  3. MySql系列之单表查询

    单表查询的语法 SELECT 字段1,字段2... FROM 表名 WHERE 条件 GROUP BY field HAVING 筛选 ORDER BY field LIMIT 限制条数 关键字的执行 ...

  4. php后端控制可跨域的域名,允许图片跨域上传

    跨域问题经常需要面对,前端需要做的比较直接要么选择ajax异步提交,XML或者jsonp,要么表单提交前端常见跨域解决方案 jsonp基本可以搞定大部分跨域问题,但问题也比较明显,只能通过get方式提 ...

  5. Python 批处理文本文件、进行查找

    去年换了一部手机,老手机终于光荣退休了,但是里面的便签里还存有很多文字记录,这个手机还不能备份到云,只能将每个便签保留为一个个的文本文件,我想要把所有的文本文件归到一个文本文件中,手动操作太麻烦了,刚 ...

  6. ARM官方《CMSIS-RTOS教程》之线程Threads

    创建线程Creating Threads 一旦RTOS开始运行,就会有很多系统调用来管理和控制活跃的线程.默认情况下,main()函数自动被创建为第一个可运行的线程.在第一个例子里我们使用main() ...

  7. vue.2.0-路由

    vue2.0 路由: http://router.vuejs.org/zh-cn/index.html 基本使用: 1. 布局 <router-link to="/home" ...

  8. modSecurity规则学习(六)——检测模式

    传统检测模式-自主规则 传统检测模式所有规则都是“闭环”的模式.就像HTTP本身一样,单独的规则是无状态的.这意味着规则之间不共享信息,每个规则都没有关于任何先前规则匹配的信息.它仅使用其当前的单个规 ...

  9. 冒泡排序算法 C#版

    冒泡排序算法的运作如下: 1.比较相邻的元素.如果第一个比第二个大,就交换他们两个. 2.对每一对相邻元素作同样的工作,从开始第一对到结尾的最后一对.在这一点,最后的元素应该会是最大的数. 3.针对所 ...

  10. P2038 无线网络发射器选址

    题目描述 随着智能手机的日益普及,人们对无线网的需求日益增大.某城市决定对城市内的公共场所覆盖无线网. 假设该城市的布局为由严格平行的129 条东西向街道和129 条南北向街道所形成的网格状,并且相邻 ...