ForeignKey使用俩表示两张表多对一关系的外键,外键字段要定义在多属性的表中。

定义外键时,to的表可以直接写类名,但是需要注意类的定义顺序;也可以写字符串式的类名,这样就可以忽略class类的定义顺序;

class Book(models.Model):
title = models.CharField(max_length=32)
pid = models.ForeignKey('Publisher',on_delete=models.CASCADE) # 外键 on_delete属性 2.0 必填 ,1.0版本默认是models.CASCADE,建议是手动写上
 【实例】
 class Book(models.Model):
bid = models.AutoField(primary_key=True)
title = models.CharField(max_length=32)
pub = models.ForeignKey("Publisher", on_delete=models.CASCADE)

def __str__(self):
return f"{self.bid}{self.title}{self.pub}"

创建完成后会发生如图这种情况,pub自动变成了pub_id,无需惊讶,这就是它应有的样子~

那么pub与pub_id之间有什么关系呢?听经过查看pub的type可以得出结论,pub得到的是一个对象,包含pub_id在内的外键对应的对象,在django中创建外键时,默认是以目标的主键作为对应项,以下的各个操作就会利用这一点。

 在本例中外键中
pub_id == pub.pid  

查询

 all_books = models.Book.objects.all()
print(all_books)
for book in all_books:
print(book)
print(book.pk)
print(book.title)
print(book.pub,type(book.pub)) # 所关联的对象
print(book.pub_id,type(book.pub_id)) # 所关联的对象的pk
print('*' * 32)

新增

 # 方式一
models.Book.objects.create(title=title,pub=models.Publisher.objects.get(pk=pub_id))
# 方式二
models.Book.objects.create(title=title, pub_id=pub_id)

删除

 pk = request.GET.get('pk')
models.Book.objects.filter(pk=pk).delete()

编辑

 book_obj.title = title
# 修改外键列
# 方式一:
book_obj.pub_id = pub_id
# 方式二:
book_obj.pub = models.Publisher.objects.get(pk=pub_id)
book_obj.save()

外键的操作

【新建测试表结构】

from django.db import models

# Create your models here.
class Publisher(models.Model):
name = models.CharField(max_length=32)
city = models.CharField(max_length=32)

def __str__(self):
return "<Publisher object: {} {}>".format(self.id, self.name)

class Author(models.Model):
name = models.CharField(max_length=32)
age = models.IntegerField()
phone = models.CharField(max_length=11)

def __str__(self):
return "<Author object: {} {}>".format(self.id, self.name)

class Book(models.Model):
title = models.CharField(max_length=32)
publish_date = models.DateField(auto_now_add=True)
price = models.DecimalField(max_digits=5, decimal_places=2)
memo = models.TextField(null=True)
# 创建外键,关联publish
publisher = models.ForeignKey(to="Publisher",null=True, on_delete=models.CASCADE)
# 创建多对多关联author
author = models.ManyToManyField(to="Author")

def __str__(self):
return "<Book object: {} {} {} {}>".format(self.id, self.title,self.price,self.author)

跨表对象查找正向

所谓正向查找是指从建立外键的表查询目标表的查找,兄目标表查询建立外键的表就是反向查询。

 book_obj = models.Book.objects.first()
print(book_obj) # 输出book对象
print(book_obj.publisher) # book对象对应的外键对象
print(book_obj.publisher.name) # 外键对象的name属性

跨表对象查找反向

pub_obj = models.Publisher.objects.get(pk=1)
print(pub_obj) # 出版社对象
print(pub_obj.name) # 出版社名称
print(pub_obj.book_set) # 关系管理对象,与该出版社相关联的图书的关系,默认的书写方式:类名小写_set
print(pub_obj.book_set.all()) # 所关联的所有的对象

添加relate_name属性

定义反向查询时的名字,例如将Book表的publisher属性添加related_name属性,会抵消默认情况的类名小写_set

# 添加属性
publisher = models.ForeignKey(to="Publisher",null=True, on_delete=models.CASCADE,related_name='books')

# 未添加属性时,反向查询
pub_obj = models.Publisher.objects.get(pk=1)
print(pub_obj.book_set)
print(pub_obj.book_set.all())
# 添加属性后的反向查询
pub_obj = models.Publisher.objects.get(pk=1)
print(pub_obj.books)
print(pub_obj.books.all())

跨表字段查找正向

# 查询'叽叽喳喳出版社'出版的书
ret = models.Book.objects.filter(publisher__name='叽叽喳喳出版社')
print(ret.all())

跨表字段查找反向

 # 查找出版《叽叽喳喳》的出版社
ret = models.Publisher.objects.filter(book__title='叽叽喳喳')
print(ret.values())

添加related_name属性

# 添加属性
publisher = models.ForeignKey(to="Publisher",null=True, on_delete=models.CASCADE,related_name='books')
# 反向查询时
ret = models.Publisher.objects.filter(books__title='叽叽喳喳')
print(ret)

添加related_query_name属性

在字段查找时related_query_name的优先级高于related_name,但是在对象查找时,related_query_name无效

 publisher = models.ForeignKey(to="Publisher",null=True, on_delete=models.CASCADE,related_name='books',related_query_name='books1')
# 反向查询时
ret = models.Publisher.objects.filter(books1__title='叽叽喳喳')
print(ret)

关系管理对象的方法

反向查询时才能取到关系管理对象
# all
pub_obj = models.Publisher.objects.get(pk=1)
print(pub_obj.book_set.all())

# set 设置对象
pub_obj.book_set.set(models.Book.objects.filter(pk__in=[3,4]))

# add 添加对象
pub_obj.book_set.add(*models.Book.objects.filter(pk__in=[1,2]))

# remove 移除外键,前提是外键字段可以为空
pub_obj.book_set.remove(*models.Book.objects.filter(pk__in=[1,2]))

# clear 清空,前提是外键字段可以为空
pub_obj.book_set.clear(*models.Book.objects.filter(pk__in=[1,2]))

# create
pub_obj.book_set.createtitle='叽叽喳喳的世界',publisher_id=2,price=99.9)

  

Django-ORM之ForeignKey的使用-多对一关系的更多相关文章

  1. django数据库操作-增删改查-多对多关系以及一对多(外键)关系

    一.一对多(外键) 例子:一个作者对应多本书,一本书只有一个作者 model代码: class Person(models.Model); name = models.CharField('作者姓名' ...

  2. 转: django数据库操作-增删改查-多对多关系以及一对多(外键)关系

    原文链接:http://blog.csdn.net/u010271717/article/details/22044415 一.一对多(外键) 例子:一个作者对应多本书,一本书只有一个作者 model ...

  3. Django框架之第六篇(模型层)--单表查询和必知必会13条、单表查询之双下划线、Django ORM常用字段和参数、关系字段

    单表查询 补充一个知识点:在models.py建表是 create_time = models.DateField() 关键字参数: 1.auto_now:每次操作数据,都会自动刷新当前操作的时间 2 ...

  4. Django ORM 多表操作

    目录 Django ORM 多表操作 表模型 表关系 创建模型 逆向到表模型 插入数据 ORM 添加数据(添加外键) 一对多(外键 ForeignKey) 一对一 (OneToOneFeild) 多对 ...

  5. django -- 多对多关系的实现

    在django中表和表之间的多对多关系有两种实现方案: 方案一:直接使用django自动实现的多对多关系. 方案二:自己写连接表.然而告诉django在实现多对多关系时要使用的连接表. 一.方案一: ...

  6. Django ORM、一对一、一对多、多对多、详解

    上篇博客也提到这些知识点,可能大家还是不太清楚,这篇博客为大家详细讲解ORM中的几个知识点 1.1首先我们先看一个小案例: #_*_coding:utf-8_*_ from django.db imp ...

  7. Django --ORM常用的字段和参数 多对多创建形式

    1 ORM字段 AutoField int自增列,必须填入参数 primary_key=True.当model中如果没有自增列,则自动会创建一个列名为id的列. IntegerField 一个整数类型 ...

  8. django ORM模型表的一对多、多对多关系、万能双下划线查询

    一.外键使用 在 MySQL 中,如果使用InnoDB引擎,则支持外键约束.(另一种常用的MyIsam引擎不支持外键) 定义外键的语法为fieldname=models.ForeignKey(to_c ...

  9. 数据库开发-Django ORM的多对多查询

    数据库开发-Django ORM的多对多查询 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 一.联合主键问题 CREATE TABLE `employees` ( `emp_no` ...

随机推荐

  1. ChromePassword

    # -*- coding: utf-8 -*- 2# @Author : pwf 3 4# @Date : 2019/5/18 22:53 5# Software : PyCharm 6# versi ...

  2. 解决使用vue打包时vendor文件过大或者是app.js文件很大的问题

    这篇文章主要介绍了使用vue打包时vendor文件过大或者是app.js文件很大问题的解决方法,非常不错,具有一定的参考借鉴价值,需要的朋友可以参考下 第一次使用vue2.0开发,之前都是用的angu ...

  3. Vue双向绑定的实现原理系列(四):补充指令解析器compile

    补充指令解析器compile github源码 补充下HTML节点类型的知识: 元素节点 Node.ELEMENT_NODE(1) 属性节点 Node.ATTRIBUTE_NODE(2) 文本节点 N ...

  4. laravel博客中文章删除遇到问题

    SQLSTATE[42S22]: Column not found: 1054 Unknown column 'blog_article.id' in 'where clause' (SQL: sel ...

  5. jquery button选择器 语法

    jquery button选择器 语法 作用::button 选择器选取类型为 button 的 <button> 元素和 <input> 元素.大理石平台价格表 语法:$(& ...

  6. java怎样实现重载一个方法

    重载(重新载选方法): java允许在一个类中,存在多个方法拥有相同的名字,但在名字相同的同时,必须有不同的参数,这就是重载,编译器会根据实际情况挑选出正确的方法,如果编译器找不到匹配的参数或者找出多 ...

  7. ueditor编辑器+粘贴word

    最近公司做项目需要实现一个功能,在网页富文本编辑器中实现粘贴Word图文的功能. 我们在网站中使用的Web编辑器比较多,都是根据用户需求来选择的.目前还没有固定哪一个编辑器 有时候用的是UEditor ...

  8. TTTTTTTTTTTTTT hdu 5763 Another Meaning 哈希+dp

    Another Meaning Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Others)T ...

  9. 删除git的远程分支

    命令:$ git push origin [空格][冒号][需要删除的分支名字] git push origin :dev 删除git的凭证的方法: 控制面板\用户帐户\凭据管理器  --->普 ...

  10. Centos-Redhat下远程桌面的方法 & Redhat改Centos源

    折腾了好几天才搞定,Redhat下远程桌面的方法,首先保证本身已经装了桌面,并且可以ssh访问 由于系统中自带python2环境,装了anaconda以及它带的python3环境,这个必须存在(前提) ...