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
猜你喜欢
- bytes 和 bytearraybytes 和 bytearray 都是二进制世界的成员,用二进制的方式去理解才能看清他的本质。理解byt
- <%DIM LinkArray(4,2)' 定义链接数组LinkArray(0,1)=&quo
- 今天偶然看到“一个有将近两年的div + CSS 开发经验和历史,曾经是Web标准绝对拥趸的同志”在自己的blog上发表放弃div+css的
- 本文实例讲述了Python判断Abundant Number的方法。分享给大家供大家参考。具体如下:Abundant Number,中文译成
- 本文实例讲述了PHP基于迭代实现文件夹复制、删除、查看大小等操作的方法。分享给大家供大家参考,具体如下:前面一篇 PHP递归实现文件夹的复制
- 写在前面 最近和几个小伙伴们在写字节跳动第五届青训营后端组的大作业。虽然昨天已经提交了项目,但有很多地方值得总结一下,比如这一篇,
- javascript中的数组对象捆绑了强大的方法因此它可以用很简短的代码实现强大的数组操作而这些功能要C或者c++实现的话可能需要花费几倍的
- SQL语句参考及记录集对象详解1. ASP与Access数据库连接:2. ASP与SQL数据库连接:建立记录集对象:set rs=serve
- 阅读上一章:Chapter 13 为文字指定样式Chapter 14 图片替换随着更多设计师与开发者开始使用标准(特别是CSS),每天都会有
- 本文实例讲述了Python常见MongoDB数据库操作。分享给大家供大家参考,具体如下:MongoDB 是一个基于分布式文件存储的数据库。由
- 不同于行级或页级锁定的选项:· 版本(例如,为并行的插入在MySQL中使用的技术),其中可以一个写操作,同时有许多读取操作。这明数据库或表支
- 应用目标:制作文字特效 应用软件:Dreamweaver MX操作难度:★★☆☆☆我们常用的网页制作工具Dreamweaver MX不仅可以
- isset(PHP 3, PHP 4, PHP 5 )isset -- 检测变量是否设置描述bool isset ( mixed var [
- yaml简介1.yaml [ˈjæməl]: Yet Another Markup Language :另一种标记语言。
- 因为主键可以唯一标识某一行记录,所以可以确保执行数据更新、删除的时候不会出现张冠李戴的错误。当然,其它字段可以辅助我们在执行这些操作时消除共
- 本文实例讲述了PHP判断是否微信访问的方法。分享给大家供大家参考,具体如下:在开发中有时需要禁止或者仅允许微信浏览器进行访问,则此时就需要对
- 本文实例讲述了Go语言通过Luhn算法验证信用卡卡号是否有效的方法。分享给大家供大家参考。具体实现方法如下:package mainimpo
- 问:握怎样测试 MySQL安装?答:可以通过以下命令测试MySQL服务器是否工作:C:\> C:\Program Files\MySQ
- 1. 使用os.system("cmd")特点是执行的时候程序会打出cmd在Linux上执行的信息。import oso
- 1、字符串的定义所谓字符串,就是由0个或者多个字符组成的有限序列。在Python程序中,如果我们把单个或多个字符用单引号''