模型之间可以有三种表关系,即一对一,一对多和多对多。表关联之间的数据操作在Django中可以很方便的操作到。在模型中,表关联的字段类型是关联表的实例,而不是字段本身类型。关联字段在数据库中会在其后补上_id,这才是关联字段本身的类型。这句话听起来很绕,下面具体来看看。

下面是学生和学院的表模板。

class Student(models.Model):#学生表
s_id = models.AutoField(primary_key=True)
s_name = models.CharField(max_length=30)
department = models.ForeignKey('Department',on_delete=models.CASCADE,related_name = 'student',null=True)
course = models.ManyToManyField('Course',related_name='student') def __str__(self):
return ('Student<id = {},name={},department={}>'.format(self.s_id, self.s_name,self.department_id)) class Department(models.Model): #学院表
d_id = models.AutoField(primary_key=True)
d_name = models.CharField(max_length=15)
def __str__(self):
return('Department<id = {},name={}>'.format(self.d_id,self.d_name))

下面我们为了方便操作,在项目目录下输入python manage.py shell进入交互模式方便操作查看。进入后将所创建的模板都导进去,输入from music.models import Department,Student,Course,Stu_detail 。

  • 下面以第一种方式来添加数据。
>>> s3 = Student(s_name = '小松',department_id = 1)
>>> s3
<Student: Student<id = None,name=小松,department=1>>
>>> s3.save()
>>>

看到上面会有疑问,Student类中的学院属性是department,为什么在实例化会是department_id呢?上面说过,对于关联字段,数据库会自动在其后加上_id,对于int类型是department_id。而关联表实例对象才是department的类型。如果将上述代码的department_id改成department将会报错。

>>> s3 = Student(s_name = '小松',department = 1)
Traceback (most recent call last):
File "<console>", line 1, in <module>
File "/home/pyvip/.virtualenvs/Django2.0/lib/python3.5/site-packages/django/db/models/base.py", line 467, in __init__
_setattr(self, field.name, rel_obj)
File "/home/pyvip/.virtualenvs/Django2.0/lib/python3.5/site-packages/django/db/models/fields/related_descriptors.py", line 210, in __set__
self.field.remote_field.model._meta.object_name,
ValueError: Cannot assign "1": "Student.department" must be a "Department" instance.
  • 第二种方式添加数据
d2 = Department(d_name = '艺术学院')
>>> d2.save()
>>> d2
<Department: Department<id = 2,name=艺术学院>>
>>> s4 = Student(s_name = '小花')
>>> s4.department = d2
>>> s4.save()
>>> s4
<Student: Student<id = 4,name=小花,department=2>>

先创建一个艺术学院,再实例化后点出属性赋值一个学院的实例化。

表关联对像的访问

对于类模板中已有的关联字段,可以直接正向查询得到,如学生的学院表,可以直接点出其department。但对于未有的关联字段,如查询学院表的学生,就无法直接来查询,因为其没有学生这个关联属性。我们可以通过反向查询,即通过小写的关联表名加上_set来查询。

>> s4.department
<Department: Department<id = 2,name=艺术学院>>
>>> d1.student_set.all()
<QuerySet [<Student: Student<id = 1,name=小杰,department=1>>, <Student: Student<id = 2,name=小王,department=1>>, <Student: Student<id = 3,name=小松,department=1>>]>

可以在关联表时加上参数related_name = 小写本表名 ,来取代foo_set。

>>> d1.student.all()
<QuerySet [<Student: Student<id = 1,name=小杰,department=1>>, <Student: Student<id = 2,name=小王,department=1>>, <Student: Student<id = 3,name=小松,department=1>>]>

表关联对象的一些方法

表关联对象除了一对一及一对多中一的这一方,其他的表关系的对象实则是一个管理器对象。如下面的d1.student,这就是一个管理器对象。

>>> d1.student
<django.db.models.fields.related_descriptors.create_reverse_many_to_one_manager.<locals>.RelatedManager object at 0xb639416c>

下面增加一个课程表。

class Course(models.Model): #课程表
c_id = models.AutoField(primary_key=True)
c_name = models.CharField(max_length=15)
def __str__(self):
return('Course<id = {},name={}>'.format(self.c_id,self.c_name))

下面创建一们python课程,通过课程反向查询学生,可以发现这也是一个管理器对象。

>>> c1 = Course(c_name = 'python')
>>> c1.save()
>>> c1.student
<django.db.models.fields.related_descriptors.create_forward_many_to_many_manager.<locals>.ManyRelatedManager object at 0xb6372a0c>

通过学生查其上的课程也是一样的。

>>> s4.course
<django.db.models.fields.related_descriptors.create_forward_many_to_many_manager.<locals>.ManyRelatedManager object at 0xb637282c>

下面再加入一个与学生一对一的学生详情表。

class Stu_detail(models.Model): #学生详细表
stu_id = models.OneToOneField('Student',on_delete=models.CASCADE)
age = models.IntegerField()
gender = models.BooleanField(default=1) #默认值1表示男性
city = models.CharField(max_length=15)
def __str__(self):
return ('Stu_detail<id={},age={},gender={},city={}'.format(self.stu_id,self.age,self.gender,self.city))

下面创建一个学生详情的数据,分别通过学生查其详情和通过详情查学生,可以发现其是一个实例,不是一个管理器对象。

ss1 = Stu_detail(stu_id_id = 1,age = 18,gender = 1,city='南昌')
>>> ss1
<Stu_detail: Stu_detail<id=Student<id = 1,name=小杰,department=1>,age=18,gender=1,city=南昌>
>>> ss1.save()
>>> ss1.stu_id
<Student: Student<id = 1,name=小杰,department=1>>
s1.stu_detail
<Stu_detail: Stu_detail<id=Student<id = 1,name=小杰,department=1>,age=18,gender=True,city=南昌>

虽然一对一时,学生类中没有学生详情表的关联字段,但可以直接点出其小写表名,便可查询。

综上可总结出,有管理器对象的是一对多的多和多对多的两个多。管理器有着属于自己的四种常用方法。下面来介绍下:

  • add()方法

add方法可以把已经存在数据库中的模型实例加入到关联对象中去。如创建一个学生,但不指定其所属学院,最后通过以学院的add方法来给其传入一个刚刚创建的学生实例,达到将其添加到指定学院中去。

>>> s5 = Student(s_name = '小龙')
>>> s5.save()
>>> s5
<Student: Student<id = 5,name=小龙,department=None>>
>>> d2.student.add(s5)
>>> s5
<Student: Student<id = 5,name=小龙,department=2>>
>>>
  • create()方法

与add方法不同的是,create方法传入的数据不需要是已存在的。它是添加一个新数据,并将其存入数据库。

>>> d1.student.create(s_name = '小李')
<Student: Student<id = 6,name=小李,department=1>>
  • remove()方法

remove移除,指定将某一实例化的某一关联对象移除掉。如下面将一个学生从一个学院中开除掉。

>>> d2.student.remove(s5)
>>> Student.objects.get(s_id =5)
<Student: Student<id = 5,name=小龙,department=None>>
  • clear()方法

清空方法,可以清空某一实例化的某一个关联字段的所有对象。如将一学院的所有学生都清空掉。

>>> d1.student.all()
<QuerySet [<Student: Student<id = 1,name=小杰,department=1>>, <Student: Student<id = 2,name=小王,department=1>>, <Student: Student<id = 3,name=小松,department=1>>, <Student: Student<id = 6,name=小李,department=1>>]>
>>> d1.student.clear()
>>> d1.student.all()
<QuerySet []>

多表查询

关联表之间可以互相作为桥梁,充当查询条件来查询到指定的值。其基本语法是:正向查询——表名.objects.filter(字段名__字段名 条件语句)  反向查询——表名.objects.filter(小写表名__字段名 条件语句)

下面是正向查询,查询一个学生id为1的学生的详情表

>>> Stu_detail.objects.filter(stu_id__s_id = 1)
<QuerySet [<Stu_detail: Stu_detail<id=Student<id = 1,name=小杰,department=1>,age=18,gender=True,city=南昌>]>

下面是一个反向查询的例子,查询一个名字中带王的所属学院。

>>> Department.objects.filter(student__s_name__contains='王')
<QuerySet [<Department: Department<id = 1,name=信息学院>>]>

聚合查询

通过关键字aggregate可以执行求最大值Amx、最小值Min、计数Count、求和Sum、平均值Avg等。语法规则是:表名.objects.all()aggregrate(Max('字段名'))  ,其返回的是一个字典的类型。在使用聚合查询前,要先记得导包。返回的字典的key部分是django自动生成的‘字段名__函数名',我们也可以自己去一个别名来取代key,即采用赋值的方式,表名.objects.all()aggregrate(别名=Max('字段名'))

>>> from django.db.models import Count,Max,Min,Sum
>>> Student.objects.all().aggregate(Max('s_id'),Min('s_id'),Sum('s_id'),Count('s_id'))
{'s_id__sum': Decimal('21'), 's_id__max': 6, 's_id__min': 1, 's_id__count': 6}

Q查询

Q查询是一个多条件的查询,Q条件之间通过‘&’、‘|’、’~、来分别表示与、或、非。用Q查询前同样也要导包。

from django.db.models import Q

>>> Student.objects.filter(Q(s_id  =1)|Q(s_name__contains='王'))
<QuerySet [<Student: Student<id = 1,name=小杰,department=1>>, <Student: Student<id = 2,name=小王,department=1>>]>
>>> Student.objects.filter(Q(s_id__lte=4)&Q(s_name__contains='王'))
<QuerySet [<Student: Student<id = 2,name=小王,department=1>>]>

F查询

F查询是一个多字段值进行比较的查询。

from django.db.models import F

>>> Student.objects.filter(department__d_id__lt=F('s_id'))
<QuerySet [<Student: Student<id = 2,name=小王,department=1>>, <Student: Student<id = 3,name=小松,department=1>>, <Student: Student<id = 4,name=小花,department=2>>, <Student: Student<id = 5,name=小龙,department=2>>, <Student: Student<id = 6,name=小李,department=1>>]>

上述查询学生所属学院id小于等于其学生id的所有学生。

分组查询

分组查询的关键字是annotate,下面的例子第一个values的作用是以什么进行分组,第二个values的作用是以什么字段输出。

>>> Student.objects.all().values('department').annotate(count = Count('department')).values('department_id','count')
<QuerySet [{'department_id': 1, 'count': 4}, {'department_id': 2, 'count': 2}]>

Django模型基础(三)——关系表的数据操作的更多相关文章

  1. Django 08 Django模型基础3(关系表的数据操作、表关联对象的访问、多表查询、聚合、分组、F、Q查询)

    Django 08 Django模型基础3(关系表的数据操作.表关联对象的访问.多表查询.聚合.分组.F.Q查询) 一.关系表的数据操作 #为了能方便学习,我们进入项目的idle中去执行我们的操作,通 ...

  2. Django——8 关系表的数据操作 表关联对象的访问 多表查询

    Django 关系表中的数据操作 表关联对象的访问 关联对象的add方法 create方法 remove方法 clear方法 多表查询 查询补充 聚合查询 分组查询 F查询 Q查询 关系表的数据操作 ...

  3. 九.django模型基础(三)之关联对象操作及多表查询

    Ⅰ.关系表的数据操作 1.正向 正向:如果一个模型有外键字段,通过这个模型对外键进行操作叫做正向. 1)更新(增) a.通过属性复制 b.通过主键的方式 总结: ForeignKey 字段的更新,跟普 ...

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

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

  5. Django基础五之django模型层(二)多表操作

    一 创建模型 表和表之间的关系 一对一.多对一.多对多 ,用book表和publish表自己来想想关系,想想里面的操作,加外键约束和不加外键约束的区别,一对一的外键约束是在一对多的约束上加上唯一约束. ...

  6. Django 07 Django模型基础2 (常用查询和多表关联)

    Django 07 Django模型基础2 (常用查询和多表关联) 一.常用查询 #查找数据 def search_user(request): #获取 rs = User.objects.first ...

  7. Django 06 Django模型基础1(ORM简介、数据库连接配置、模型的创建与映射、数据的增删改查)

    Django 06 Django模型基础1(ORM简介.数据库连接配置.模型的创建与映射.数据的增删改查) 一.ORM系统 #django模型映射关系 #模型类-----数据表 #类属性-----表字 ...

  8. 06.Django基础五之django模型层(二)多表操作

    一 创建模型 表和表之间的关系 一对一.多对一.多对多 ,用book表和publish表自己来想想关系,想想里面的操作,加外键约束和不加外键约束的区别,一对一的外键约束是在一对多的约束上加上唯一约束. ...

  9. day 70 Django基础五之django模型层(二)多表操作

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

随机推荐

  1. P1054 求平均值

    P1054 求平均值 转跳点:

  2. Python基础笔记:函数式编程:高阶函数、返回函数、匿名函数

    高阶函数 高阶函数:一个函数可以接收另一个函数作为参数 或 一个函数可以返回一个函数作为返回值,这种函数称之为高阶函数. #函数 add 接收 f 函数作为参数 >>> def ad ...

  3. 74.Python中ORM聚合函数详解:Max,Min

    Max和Min:获取指定对象的最大值和最小值. 1. 比如:想要获取Author表中的最大的年龄和最小的年龄.示例代码如下: from django.http import HttpResponse ...

  4. css怎么让图片垂直左右居中?(外层div是浮动且按照百分比排列)

    一.原始的居中方法是把div换成table <div style="width: 500px; height: 200px; border: solid 1px red; text-a ...

  5. 腾讯云服务器上搭建Jenkins配置邮箱通知

    1,Jenkins 点击 系统管理 2,点击系统管理 3,配置系统管理员邮件地址 5,配置 Extended E-main Notification,(用户名不需要邮箱后缀“@163.com”, SS ...

  6. [ WARN ] Keyword 'Capture Page Screenshot' could not be run on failure: URLError: <urlopen error [Errno 10061] Connection refused>

    [ WARN ] Keyword 'Capture Page Screenshot' could not be run on failure: URLError: <urlopen error ...

  7. Day 22:网络编程(3)

    TCP通讯协议特点:  1. tcp是基于IO流进行数据 的传输的,面向连接.  2. tcp进行数据传输的时候是没有大小限制的.  3. tcp是面向连接,通过三次握手的机制保证数据的完整性.可靠协 ...

  8. 小程序实现倒计时:解决ios倒计时失效(setInterval失效)

    在使用之前需要先在page页引入wxTimer.js文件(这里我将文件放在/utils) let timer = require('../../utils/wxTimer.js'); 然后就可以使用啦 ...

  9. MFC 屏蔽esc跟enter键

    BOOL CMenuOperate::PreTranslateMessage(MSG* pMsg) { if(pMsg->message == WM_KEYDOWN && pMs ...

  10. (7)opencv图片内部的基本处理

    就是,给定我们一张图片,我们可以对图片的每一个像素的色彩进行处理 比如,我们的原图是这个样子 然后我首先将他变成灰度图(灰度图的行道是1,就是chanaual是1) 然后,我又将灰色图片的黑白进行颠倒 ...