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

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. 王立平--scard0与scard1分别指的是什么?以及路径获取

    一般是: scard0:指系统内部存储 scard1:指外插的sd卡 也有特例.. 分别获取路径的方法: package com.main; import java.lang.reflect.Meth ...

  2. 【Excle】使用&拼接,拼接后复制到文本编辑器存在引号(“”)问题

    1.错误的操作 如下图:拼接 拼接后,复制到文本编辑器: 1.1错误原因 出现上述问题的原因:是由于在拼接字符串的时候, INSERT INTO CST(ID,NAME)values( 上述这段字符& ...

  3. Android RxJava使用介绍(四) RxJava的操作符

    本篇文章继续介绍下面类型的操作符 Combining Observables(Observable的组合操作符) Error Handling Operators(Observable的错误处理操作符 ...

  4. [Material Design] 教你做一个Material风格、动画的button(MaterialButton)

    原创作品,转载请注明出处:http://blog.csdn.net/qiujuer/article/details/39831451 前段时间Android L 公布了,相信看过公布会了解过的朋友都为 ...

  5. 源代码分析:onAttach, onMeasure, onLayout, onDraw 的顺序。

    从前文<源代码解析:dialog, popupwindow, 和activity 的第一个view是怎么来的?>中知道了activity第一个view或者说根view或者说mDecorVi ...

  6. Unity2D实现人物三连击

    之前写过一个系列<HTML5 2D平台游戏开发>,在此过程中发现有很多知识点没有掌握,而且用纯JavaScript来开发一个游戏效率极低,因为调试与地图编辑都没有可视化的工具,开发起来费时 ...

  7. 不同手机根据坐标计算控件、图片的像素,px 与 dp, sp换算公式?

    参考该帖子:http://www.cnblogs.com/bluestorm/p/3640786.html PPI = Pixels per inch,每英寸上的像素数,即 "像素密度&qu ...

  8. spring download

    http://maven.springframework.org/release/org/springframework/spring/

  9. linux系统寻找僵尸进程

    1. 用top命令来查看服务器当前是否有僵尸进程. 2. 用ps和grep命令寻找僵尸进程 $ ps -A -ostat, pid, ppid, cmd | grep -e '^[Zz]' 命令解释: ...

  10. Hadoop2.6.0版本MapReudce示例之WordCount(一)

    一.准备测试数据 1.在本地Linux系统/var/lib/Hadoop-hdfs/file/路径下准备两个文件file1.txt和file2.txt,文件列表及各自内容如下图所示: 2.在hdfs中 ...