django ManyToManyField多对多关系的实例详解
作者:zenge_blog 发布时间:2023-08-11 20:21:01
Django 的 ORM 有多种关系:一对一,多对一,多对多
各自定义的方式为 :
一对一: OneToOneField
多对一: ForeignKey
多对多: ManyToManyField
上边的描述太过数据而缺乏人性化,我们来更人性化一些:
多个属于一个,即 belong to : ForeignKey,多个属于一个
一个有一个,即 has one: OneToOneField
一个有很多个,即 has many: lots of A belong to B 与 B has many A,在建立 ForeignKey 时,另一个表会自动建立对应的关系
一个既有很多个,又属于很多个,即 has many and belong to : ManyToManyField,同样只能在一个model类中说明,关联表会自动建立。
多对多的关系:
举例:现有两张表,user表和group表。user表中的字段有用户名,邮箱,年龄等信息。而group表中有组名信息。我们知道一个用户可以属于多个组,一个组中也可以包含多个用户,所以这两张表的关系就是多对多的关系。
mysite/learn/models.py文件代码
#coding:utf8
from django.db import models
class Group(models.Model):
Name = models.CharField(max_length=20)
def __unicode__(self):
return self.Name
class User(models.Model):
Name = models.CharField(max_length=20)
Email = models.CharField(max_length=50)
group = models.ManyToManyField(Group,blank=True)
def __unicode__(self):
return self.Name
def group_list(self):
return ','.join([i.Name for i in self.group.all()])
创建两张表Group和User,Group表中只有组名“Name”这一个字段。而User表中有用户名“Name”,邮箱“Email”,组名“group”三个字段。
在User表中,由于group信息与Group表关联的,所以要在User表中设置
ManyToManyField
def group_list(self):
return ','.join([i.Name for i in self.group.all()])
定义group_list函数,是为了在后台页面中显示group_list字段信息。group_list是后台页面显示的字段名称。
i.Name for i in self.group.all()
这里Name是Group表中的Name字段,self.group中的group是User表自己的group字段
mysite/admin.py文件中的代码
from django.contrib import admin
from learn.models import *
# Register your models here.
class UserAdmin(admin.ModelAdmin):
list_display = ['id','Name','Email','group_list']
admin.site.register(User,UserAdmin)
class GroupAdmin(admin.ModelAdmin):
list_display = ['id','Name']
admin.site.register(Group,GroupAdmin)
访问admin后台管理页面
在group表中创建组
在user表中创建用户,Group字段选择用户组。
仔细看上面的截图,会发现一个问题-->在Group表中只有组名字段,但是看不到每个组中都有哪些用户。而User表中可以看到group_list字段,所以如果希望在Group中显示用户信息,可以仿照User表的做法,mysite/learn/models.py文件代码
#coding:utf8
from django.db import models
class Group(models.Model):
Name = models.CharField(max_length=20)
def user_list(self):
return ','.join([i.Name for i in self.user_set.all()])
def __unicode__(self):
return self.Name
class User(models.Model):
Name = models.CharField(max_length=20)
Email = models.CharField(max_length=50)
group = models.ManyToManyField(Group,blank=True)
def __unicode__(self):
return self.Name
def group_list(self):
return ','.join([i.Name for i in self.group.all()])
在learn/admin.py文件中加上user_list字段
再次访问admin后台管理页面,在group表中可以看到user_list信息了。
正向查询和反向查询
正向查询:
上面我们创建了两张表user和group,现在我想查询user表中某个用户的所属组
进入django shell命令行
python manage.py shell
>>> from learn.models import *
>>> User.objects.all()
[<User: 老黄>, <User: 老张>, <User: 老王>]
>>> User.objects.all()[0]
<User: 老黄>
>>> User.objects.all()[0].Email
u'laohuang@qq.com'
>>> User.objects.all()[0].group.all()
[<Group: CEO>, <Group: COO>]
>>> User.objects.all()[0].group.all()[0].Name
u'CEO'
>>> User.objects.all()[0].group.all()[0].id
mysite/models.py文件中的每一个类都是一个对象,使用
User.objects.all()
可以获取所有对象,它是一个列表
[<User: 老黄>, <User: 老张>, <User: 老王>]
获取第一个对象
>>> User.objects.all()[0]
<User: 老黄>
获取老黄这个对象的邮箱属性的值
>>> User.objects.all()[0].Email
u'laohuang@qq.com'
获取用户所属组的组名,和id
>>> User.objects.all()[0].group.all()[0].Name
u'CEO'
>>> User.objects.all()[0].group.all()[0].id
反向查询:
>>> from learn.models import * ##导入models中所有的类
>>> Group.objects.all() ##查看Group表中所有的对象
[<Group: CEO>, <Group: CTO>, <Group: COO>, <Group: VP>]
>>> Group.objects.all()[0] ##查看Group表中第一个对象CEO
<Group: CEO>
>>> Group.objects.all()[0].Name ##查看CEO这个对象的Name属性
u'CEO'
>>> Group.objects.all()[0].user_set.all() ##反向查看CEO这个对象的用户名
[<User: 老黄>]
>>> Group.objects.all()[0].user_set.all()[0]
<User: 老黄>
>>> Group.objects.all()[0].user_set.all()[0].Email ##反向查看CEO这个对象的Email
u'laohuang@qq.com'
>>> Group.objects.all()[0].user_set.all()[0].Name
u'\u8001\u9ec4'
来源:https://blog.51cto.com/zengestudy/1902277


猜你喜欢
- 你可以按照以下方法使用 ls 命令来查看你的系统中都有那些 Python 的二进制文件可供使用。$ ls /usr/bin/python*/
- 1、一个或多个文件夹组成一个模块,而一个模块组合构成了一个包发布在公共目录里。2、包必须有__init__文件,否则就是一个文件夹。实例im
- 去年淘宝做了个“胖子”项目,就是把网页的默认宽度从780提升到了950。也就是说,基本放弃了800×600的用户(没有完全放弃,如果你仔细研
- 本文实例讲述了python实现将元祖转换成数组的方法。分享给大家供大家参考。具体分析如下:python的元祖使用一对小括号表示的,元素是固定
- 问题描述现有一个有向赋权图。如下图所示:问题:根据每条边的权值,求出从起点s到其他每个顶点的最短路径和最短路径的长度。说明:不考虑权值为负的
- 这篇文章主要介绍了python基于property()函数定义属性,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价
- 开放平台的API接口调用需要限制其频率,以节约服务器资源和避免恶意的频繁调用使用自定义频率限制组件:utils/thottle.pyclas
- 由于该math模块与 Python 版本一起打包,因此您不必单独安装它,直接导入:import mathmath模块常数Pythonmath
- 数据结构channel的数据结构在$GOROOT/src/runtime/chan.go文件下:type hchan struct {qco
- 目录1. 配置Python环境变量2. 安装Python编辑器,并在其中配置Python3. 安装控制包uiautomator2,和其它辅助
- 效果如下:代码如下:<!DOCTYPE html><html> <head> <meta char
- [sql] -- ===================【创建存储过程】===================== USE [Message
- 面对缺失值三种处理方法:option 1: 去掉含有缺失值的样本(行)option 2:将含有缺失值的列(特征向量)去掉option 3:将
- Vue页面、组件之间传参方式繁多,此处罗列出常用的几种方式,欢迎审阅补充。一丶路由传参这里的路由传参以编程式 router.push(...
- 写文的原因学习 Python 的过程中,经常会使用 pip 命令去安装第三方模块,但细细琢磨下来,我真的掌握这款工具了吗?本文为你全面揭示一
- 二分查找算法:简单的说,就是将一个数组先排序好,比如按照从小到大的顺序排列好,当给定一个数据,比如target,查找target在数组中的位
- 如何要恢复Master数据库呢?方法1:重装SQL,但是数据就Over掉了方法2:重建Master Rebuildm.exe 用到SQL的安
- 列表生成式基础语法[exp for iter_var in iterable (if conditional)]原理:首先迭代 iterab
- 1. 修改pip install默认安装路径一般使用Anaconda时会使用pip install ###来安装各类包,但默认安装路径在C盘
- MySQL有时候忘记了root密码是一件伤感的事。这里提供Windows 和 Linux 下的密码重置方法。Windows:1.以系统管理员