Django 多对多字段的更新和插入数据实例
作者:lmw1239225096 发布时间:2022-09-11 01:17:04
表的普通字段 一对多字段 多对多字段 插入数据
#插入数据
def add(request):
G_title=request.POST.get('title')#-------值为:python书本
G_publish=request.POST.get('publish') #-------值为:1
G_authors_list=request.POST.getlist('authors')#-------值为:[3,7]
publish_obj=Publish.objects.get(id=G_publish)#查找Publish表对应id的obj
authors_obj_list=Author.objects.filter(id__in=G_authors_list)#查找Author表对应id的多个obj
title #-------普通字段
publish #-------一对多外键
authors #-------多对多外键
book_obj=Book.objects.create(title=G_title,publish=publish_obj)#添加普通和一对多外键的值
#添加多对多外键的值
方式一
book_obj.authors.add(*authors_obj_list)
方式二
for obj in authors_obj_list:
book_obj.authors.add(obj)
return redirect('/index/')
表的普通字段 一对多字段 多对多字段 修改数据
def edit(request,b_id):#b_id-----为书本的id
G_title=request.POST.get('title')#-------值为:linux书本
G_publish = request.POST.get('publish')#-------值为:2
G_authors_list = request.POST.getlist('authors')#-------值为:[3,5]
book_obj = Book.objects.get(id=b_id)#查找Book表要修改的id对象
publish_obj = Publish.objects.get(id=G_publish)#查找Publish表对应id的obj
author_objs = Author.objects.filter(id__in=G_authors_list)#查找Author表对应id的多个obj
book_obj.title=G_title #-------修改普通字段
book_obj.publish=publish_obj #-------修改一对多字段
book_obj.authors=author_objs #-------修改多对多字段
#注意:多对多字段 必须是.all()/.filter()等的查询集(QuerySet)
book_obj.save() #-------一定得save(),才能对数据库进行修改
return redirect('/index/')
补充知识:Django的ManyToManyField(多对多)的使用以及through的作用
创建一个经典的多对多关系:一本书可以有多个作者,一个作者可以有多本书(如下)
进行数据迁移,然后我们使用python manage.py sqlmigrate app(应用名) 迁移文件名 查看一下sql语句(如下):
从图可以看出生成了三张表,一个是book(书籍)表包含id,title两个字段,一个是author(作者表)包含id,name,email三个字段,这是我们刚刚在models.py文件中创建两个模型,但是有一点需要注意的是在book表里面没有我们创建的authors表,而是多了一个book_authors表,在这张表里面又多了两个字段book_id,author_id,其实这个第三张表就是用来存放书籍和作者之间映射关系的中间表
那么我们如何进行数据的查询呢?
1.一本书的所有作者
b = Book.objects.get(id=1)
b.author.all()
2.一个作者的所有书籍:
a = Author.objects.get(id=1)
a.book_set.all()
3.给多对多的字段添加值(添加多对多关系):
从多对多字段中删除值(删除多对多关系):
可以看出这个blog_book_authors是根据多对多关系自动生成的关系表,但是如果我们想要搜集关于这个作者发布某一本书籍的时间额外增加一个字段,或者说与现有的系统集成,这个关系表已经存在了,那对于这样的情形,Django允许指定一个用于管理多对多关系的中间模型,然后就可以把这些额外的字段添加到这个中间模型中,具体的方法就是在ManyToMany字段中指定through参数指定作为中介的中间模型,修改上述models.py:
查看一下此时的数据库表结构:
在创建BookAuthor类的最后使用了自定义的名称,这个也可以不用指定,系统可以自动生成表名
仔细观察BookAuthor这个类,也就是我们前面讲到的中间模型,同时我们看到在创建中间模型的时候我们创建了两个外键,这两个外键定义了两个模型之间是如何关联到一起的
所以当创建多对多关系模型的时候提倡使用through参数去指定并创建中间模型,这样比较方便我们进行字段的扩展
那么此时我们又该如何添加和删除多对多关系呢?还能使用和刚刚相同的方法吗?
# 添加作者 ringo
ringo = Author.objects.create(name='Ringo',email='ringo@qq.com')
# 添加作者paul
paul = Author.objects.create(name='Paul',email='paul@qq.com')
# 添加书籍 python book1
book1 = Book.objects.create(title='python book1')
# 给多对多添加值也就是添加多对多关系
m1 = BookAuthor(author=ringo,book=book)
# 第二种添加方式
m2 = BookAuthor.objects,create(author=paul,book=book1)
当我们使用多对多的中间模型之后,add(),remove(),create()这些方法都会被禁用,所以在创建这种类型的关系的时候唯一的方法就是通过创建中间模型的实例
来源:https://blog.csdn.net/lmw1239225096/article/details/78397847


猜你喜欢
- 最近学习了python基础,写一下3大排序练练手:'''Created on 2013-8-23@author: c
- 于是写测试程序。。。不行 下载最新的ODBC。。。还是不行 通过sql plus查询。。。咦?竟然也查不到。。。 于是,折腾。。。折腾。。。
- 本文实例为大家分享了python发送邮件的具体代码,供大家参考,具体内容如下#!/usr/bin/env python # -*- codi
- 数据库事务隔离级别数据库事务的隔离级别有4个,由低到高依次为Read uncommitted:允许脏读。Read committed: 防止
- 这是工作期间同事想要个截完图之后可以显示并且永远前置的截图小工具(即不会被其他程序覆盖)直接上代码:# # -*- coding: utf-
- 这里以mysql为例,先明确以下几个问题:一.一般项目如果不自己配置事务的话,一般默认的是autocommit,即执行完一个操作后自动com
- 预测结果转为numpy:logits=model(feature)#如果模型是跑在GPU上result=logits.data.cpu().
- 很多时候我们需要让main函数不退出,让它在后台一直执行,例如:func main() { for i := 0;
- HttpRequest.FILES表单上传的文件对象存储在类字典对象request.FILES中,表单格式需为multipart/form-
- 前言这篇文章给大家讲解的是在vue-cli脚手架中如何配置vue-router前端路由的相关内容,分享出来供打击参考学习,下面来一起看看详细
- 我们经常使用nodeType==1判断元素是否是一个HMTLElement元素。页面上的元素都是节点(Node),有元素节点(Element
- private void Button1_Click(object sender, System.E
- 一 引入jwtjwt用户身份验证go get github.com/dgrijalva/jwt-go二 框架中引入jwtimport (&q
- 本文实例讲述了Python 多线程,threading模块,创建子线程的两种方式。分享给大家供大家参考,具体如下:GIL(全局解释器锁)是C
- 1在js中只有两种作用域a:全局作用域b:函数作用域在ES6之前,js是没有块级作用域。首先来解释一下什么是没有块级作用域?所以此时 是可以
- 使用一个简单的 XSL 样式表就可以将 XML 数据转换成 HTML。随着 XML 规范的不断演进,在新的版本中满足每个人的需要似乎已经成为
- 如果您在试图打开一个.MDF数据库文件时,却发现自己没有安装SQL Server数据库,该怎么办呢?这时候,如果恰巧您的机子上装有Visua
- js 日期 获取今天、昨天、明天第一个函数function getDay(day){ var today = ne
- 记得很早以前看到过这样的一段介绍:想象你在逛街边的一家书店,如果最终你没有购买任何图书就直接离开了,店长并不会知道你来过。但是如果你买了书,
- 当我们准备建立一个Web站点时,就必须向域名登记机构申请一个Internet域名,因此,我们通常希望了解自己准备使用的域名是否已经被注册,这