对于数据库来说一般表结构只会有三种对应关系,分别是一对一、一对多和多对一,下面分别介绍:

1.一对多

何为一对多,例如一个学生只可能有一个班级,一个班级却又多个学生,班级表和学生表就是一对多的关系。

在查询信息的时候有两种方式:正向查询和反向查询。

(1)正向查询

models.Student.objects.first()
<Student: Student object>#返回一个学生对象
models.Student.objects.first().sname #可以查询学生信息
'科比'
models.Student.objects.first().cid #由于有关联,可以直接获取班级对象
<Class: Class object>
models.Student.objects.first().cid.cname #查询班级属性
'全栈6期'
models.Student.objects.first().cid.first_day
datetime.date(2017, 7, 14)

(2)反向查询

models.Class.objects.filter(id=3)
<QuerySet [<Class: Class object>]>
#这是一个班级Queryset models.Class.objects.filter(id=3)[0]
<Class: Class object>
#获取班级对象 models.Class.objects.filter(id=3)[0].cname
'全栈8期'
#既然是对象就可以查看相关属性 models.Class.objects.filter(id=3)[0].student_set
<django.db.models.fields.related_descriptors.create_reverse_many_to_one_manager.<locals>.RelatedManager object at 0x0000019A6A928FD0>
#反向查找学生,获取学生集合,这是Django自己的模式 models.Class.objects.filter(id=3)[0].student_set.all()
<QuerySet [<Student: Student object>, <Student: Student object>]>
#查看集合中所有的对象 models.Class.objects.filter(id=3)[0].student_set.all().values()
<QuerySet [{'id': 8, 'sname': '小鸟', 'cid_id': 3}, {'id': 11, 'sname': '大爷', 'cid_id': 3}]>
#可以进一步查看相关属性

注意:

如果不在外键的字段中设置related_name的话,默认就用表名_set.

如果设置了related_name=“students”,反向查找时可以直接使用student进行反向查找。

>>> class_obj.students.all()
#直接获取班级中的学生,而不需要set

2.一对一

表结构设计

class Student(models.Model):
id = models.AutoField(primary_key=True)
sname = models.CharField(max_length=32)
cid = models.ForeignKey("Class")
detail = models.OneToOneField(to="StudentInfo", null=True) class StudentInfo(models.Model):
height = models.CharField(max_length=4)
weight = models.CharField(max_length=4)
addr = models.CharField(max_length=32,null=True)

(1)正向查询

models.Student.objects.first()
<Student: Student object>
#获得一个对象 models.Student.objects.first().detail.addr
'罗田'
#正向查询获取属性,通过关联字段直接找

(2)反向查询

models.StudentInfo.objects.filter(height=180)
<QuerySet [<StudentInfo: StudentInfo object>]>
#获得一个Queryset models.StudentInfo.objects.filter(height=180)[0].student
<Student: Student object>
#反向查找学生,返回一个对象 models.StudentInfo.objects.filter(height=180)[0].student.sname
'小鸟'
#查看学生信息

3.多对多

1)创建多对多的对应关系有三种方法:

(1)通过外键创建

class Class(models.Model):
id = models.AutoField(primary_key=True) # 主键
cname = models.CharField(max_length=32) # 班级名称
first_day = models.DateField() # 开班时间 class Teacher(models.Model):
tname = models.CharField(max_length=32) # 自定义第三张表,通过外键关联上面两张表
class Teacher2Class(models.Model):
teacher = models.ForeignKey(to="Teacher")
the_class = models.ForeignKey(to="Class") class Meta:
unique_together = ("teacher", "the_class") #指定联合唯一

(2)通过ManyToManyField创建

class Class(models.Model):
id = models.AutoField(primary_key=True) # 主键
cname = models.CharField(max_length=32) # 班级名称
first_day = models.DateField() # 开班时间 class Teacher(models.Model):
tname = models.CharField(max_length=32)
# 通过ManyToManyField自动创建第三张表
cid = models.ManyToManyField(to="Class", related_name="teachers")

这种方法会自动生成一张对应表,但是不能使用多对多的众多方法。

(3)通过外键和ManyToManyField创建

class Class(models.Model):
id = models.AutoField(primary_key=True) # 主键
cname = models.CharField(max_length=32) # 班级名称
first_day = models.DateField() # 开班时间 class Teacher(models.Model):
tname = models.CharField(max_length=32)
# 通过ManyToManyField和手动创建第三张表
cid = models.ManyToManyField(to="Class", through="Teacher2Class", through_fields=("teacher", "the_class")) class Teacher2Class(models.Model):
teacher = models.ForeignKey(to="Teacher")
the_class = models.ForeignKey(to="Class") class Meta:
unique_together = ("teacher", "the_class")

貌似这种方法最为靠谱,但是最麻烦。

2)查询的方法

(1)正向查询(由老师表查询班级)

models.Teacher.objects.first()
<Teacher: Teacher object>
#获取一个老师的对象 models.Teacher.objects.first().cid
<django.db.models.fields.related_descriptors.create_forward_many_to_many_manager.<locals>.ManyRelatedManager object at 0x000001FE3916A518>
#获取老师所对应的班级 models.Teacher.objects.first().cid.all()
<QuerySet [<Class: Class object>, <Class: Class object>]>
#获取所有班级对象 models.Teacher.objects.first().cid.all().values()
<QuerySet [{'id': 1, 'cname': '全栈6期', 'first_day':
datetime.date(2017, 7, 14)}, {'id': 3, 'cname': '全栈8期', 'first_day': datetime.date(2017, 10, 17)}]>
#查看所有值

为什么能够直接找到班级?

答案是:通过关联字段

(2)反向查询(由班级表反向查询老师表)

models.Class.objects.first()
<Class: Class object>
#获取班级对象 models.Class.objects.first().teachers
<django.db.models.fields.related_descriptors.create_forward_many_to_many_manager.<locals>.ManyRelatedManager object at 0x000001FE3918C390> models.Class.objects.first().teachers.all().values()
<QuerySet [{'id': 1, 'tname': '爱根'}, {'id': 4, 'tname': '日天'}, {'id': 6, 'tname': '银角大王'}]>
#由班级反向查老师 models.Class.objects.first().teachers.all()
<QuerySet [<Teacher: Teacher object>, <Teacher: Teacher object>, <Teacher: Teacher object>]>

刚才正向查找的时候,是因为老师表种由cid这个字段,所以可以直接查找,但是现在这个是怎么关联上的了?请看下面

cid = models.ManyToManyField(to="Class", related_name="teachers")

关联的时候设置了related_name,所以可以反向查找。

注意:这里不能使用teacher_set这种方式,而在一对多种可以使用

来看看各种对应关系的正向查找:

#一对多正向查找(学生表——》班级表)
models.Student.objects.first().cid #一对一正向查找(学生表——》学生信息表)
models.Student.objects.first().detail.addr #多对多正向查找(老师表到班级表)
models.Teacher.objects.first().cid.all() #正向查找是表中必须有的字段

来看看各种对应关系的反向查找:

反向查找是相对于正向查找来说的。

#一对多反向查找(班级表——》学生表)
models.Class.objects.filter(id=3)[0].student_set.all() #一对一反向查找(后面直接可查属性)(学生信息表——》学生表)
models.StudentInfo.objects.filter(height=180)[0].student #多对多反向查找(班级表——》老师表)
models.Class.objects.first().teachers.all()

Django模型系统——ORM表结构对应关系的更多相关文章

  1. Django模型系统——ORM中跨表、聚合、分组、F、Q

    核心知识点: 1.明白表之间的关系 2.根据关联字段确定正反向,选择一种方式 在Django的ORM种,查询既可以通过查询的方向分为正向查询和反向查询,也可以通过不同的对象分为对象查询和Queryse ...

  2. Django模型系统——ORM

    一.概论 1.ORM概念 对象关系映射(Object Relational Mapping,简称ORM)模式是一种为了解决面向对象与关系数据库存在的互不匹配的现象的技术. 简单的说,ORM是通过使用描 ...

  3. Django模型系统——ORM校园管理系统代码

    1.models.py from django.db import models # Create your models here. class Class(models.Model): id = ...

  4. django模型系统(一)

    django模型系统(一) djangode ORM ORM:对像关系映射 用python概念去表达数据库 数据库配置(mysql) 安装pumysql 修改项目目录下的__init__.py imp ...

  5. django模型系统(二)

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

  6. 25 Zabbix系统数据表结构介绍

    点击返回:自学Zabbix之路 点击返回:自学Zabbix4.0之路 点击返回:自学zabbix集锦 25 Zabbix系统数据表结构介绍 自学Zabbix之路15.1 Zabbix数据库表结构简单解 ...

  7. 利用flask-sqlacodegen快速导入ORM表结构

    利用flask-sqlacodegen快速导入ORM表结构 友情提示:如果是使用pymysql请预先pip install 哦~ 这是window下使用virtualenv环境下执行的 Linux用户 ...

  8. 八.django模型系统(二)之常用查询及表关系的实现

    Ⅰ.常用查询  1.几个概念 每一个django模型类,都有一个默认的管理器,objects,查询就是依赖于objects管理器进行的(在创建时就被添加了). QuerySet表示数据库中对象的列表( ...

  9. Django之django模型层一单表操作

    一 ORM简介 MVC或者MVC框架中包括一个重要的部分,就是ORM,它实现了数据模型与数据库的解耦,即数据模型的设计不需要依赖于特定的数据库,通过简单的配置就可以轻松更换数据库,这极大的减轻了开发人 ...

随机推荐

  1. javascript 数组 find

    find() 方法返回通过测试(函数内判断)的数组的第一个元素的值. let arr = [1,2,3,4] console.log(arr.find(i => {return i>1}) ...

  2. 嵌套矩形——DAG上的动态规划

    有向无环图(DAG,Directed Acyclic Graph)上的动态规划是学习动态规划的基础.非常多问题都能够转化为DAG上的最长路.最短路或路径计数问题. 题目描写叙述: 有n个矩形,每一个矩 ...

  3. javascript---》arguments对象

    使用arguments可以直接访问函数传入的实参 如:arguments[0]访问第一个参数,arguments[1]访问第二个参数 arguments.length检测函数的参数个数 如: func ...

  4. Solution to Triangle by Codility

    question: https://codility.com/programmers/lessons/4 we need two parts to prove our solution. on one ...

  5. iOS项目开发之Socket编程

    有一段时间没有认真总结和写博客了 前段时间找工作.进入工作阶段.比较少静下来认真总结,现在静下心来总结一下最近的一些心得 前言 AsyncSocket介绍 AsyncSocket详解 AsyncSoc ...

  6. xpinyin-函数返回多个值-lambda匿名函数-列表生成式-三元表达式

    import xpinyinp=xpinyin.Pinyin() #实例化print(p.get_pinyin('小白','')) 函数返回多个值:1.函数如果返回多个值的话,它会把这几个值放到一个元 ...

  7. 【SpringMVC学习10】SpringMVC对RESTfull的支持

    RESTful架构,就是目前流行的一种互联网软件架构.它结构清晰.符合标准.易于理解.扩展方便,所以正得到越来越多网站的采用.RESTful架构对url进行规范,写RESTful格式的url是什么样子 ...

  8. VS2012 未找到与约束ContractName Microsoft.VisualStudio.Text.ITextDocumentFactoryService 未找到与约束ContractName,无法打开项目的解决方案 SQLyog 注册码

    VS2012 未找到与约束ContractName Microsoft.VisualStudio.Text.ITextDocumentFactoryService   最近新换了系统还真是问题多多呀! ...

  9. Django--登录认证

    COOKIE 与 SESSION 概念 cookie的使用,不止局限于我们所使用的登录认证,cookie不属于http协议范围,由于http协议无法保持状态,但实际情况,我们却又需要“保持状态”,因此 ...

  10. 怎样新建Quartusproject—FPGA新手教程

    这一章我们来实现第一个FPGAproject-LED流水灯.我们将通过流水灯例程向大家介绍一次完整的FPGA开发流程,从新建project,代码设计,综合实现.管脚约束,下载FPGA程序. 掌握本章内 ...