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:utf8fromdjango.dbimportmodelsclassGroup(models.Model):Name=models.CharField(max_length=20)def__unicode__(self):returnself.NameclassUser(models.Model):Name=models.CharField(max_length=20)Email=models.CharField(max_length=50)group=models.ManyToManyField(Group,blank=True)def__unicode__(self):returnself.Namedefgroup_list(self):return','.join([i.Nameforiinself.group.all()])

创建两张表Group和User,Group表中只有组名“Name”这一个字段。而User表中有用户名“Name”,邮箱“Email”,组名“group”三个字段。

在User表中,由于group信息与Group表关联的,所以要在User表中设置

ManyToManyField

defgroup_list(self):return','.join([i.Nameforiinself.group.all()])

定义group_list函数,是为了在后台页面中显示group_list字段信息。group_list是后台页面显示的字段名称。

i.Nameforiinself.group.all()

这里Name是Group表中的Name字段,self.group中的group是User表自己的group字段


mysite/admin.py文件中的代码

fromdjango.contribimportadminfromlearn.modelsimport*#Registeryourmodelshere.classUserAdmin(admin.ModelAdmin):list_display=['id','Name','Email','group_list']admin.site.register(User,UserAdmin)classGroupAdmin(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:utf8fromdjango.dbimportmodelsclassGroup(models.Model):Name=models.CharField(max_length=20)defuser_list(self):return','.join([i.Nameforiinself.user_set.all()])def__unicode__(self):returnself.NameclassUser(models.Model):Name=models.CharField(max_length=20)Email=models.CharField(max_length=50)group=models.ManyToManyField(Group,blank=True)def__unicode__(self):returnself.Namedefgroup_list(self):return','.join([i.Nameforiinself.group.all()])


在learn/admin.py文件中加上user_list字段




再次访问admin后台管理页面,在group表中可以看到user_list信息了。


正向查询和反向查询

正向查询:

上面我们创建了两张表user和group,现在我想查询user表中某个用户的所属组

进入django shell命令行

pythonmanage.pyshell

>>>fromlearn.modelsimport*>>>User.objects.all()[<User:老黄>,<User:老张>,<User:老王>]>>>User.objects.all()[0]<User:老黄>>>>User.objects.all()[0].Emailu'laohuang@qq.com'>>>User.objects.all()[0].group.all()[<Group:CEO>,<Group:COO>]>>>User.objects.all()[0].group.all()[0].Nameu'CEO'>>>User.objects.all()[0].group.all()[0].id1


mysite/models.py文件中的每一个类都是一个对象,使用

User.objects.all()

可以获取所有对象,它是一个列表

[<User:老黄>,<User:老张>,<User:老王>]

获取第一个对象

>>>User.objects.all()[0]<User:老黄>

获取老黄这个对象的邮箱属性的值

>>>User.objects.all()[0].Emailu'laohuang@qq.com'

获取用户所属组的组名,和id

>>>User.objects.all()[0].group.all()[0].Nameu'CEO'>>>User.objects.all()[0].group.all()[0].id1



反向查询:

>>>fromlearn.modelsimport*##导入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这个对象的Emailu'laohuang@qq.com'>>>Group.objects.all()[0].user_set.all()[0].Nameu'\u8001\u9ec4'