Django的Many-to-Many(多对多)模型

日期:2012-05-05 |  来源:未知 |  作者:redice |  人围观 |  1 人鼓掌了!

参考:《DjangoBook2.0》 数据模型高级进阶

经典的例子:一本书有多个作者,一个作者有多本书,典型的多对多关系。

设计模型如下:

  1. from django.db import models
  2. class Author(models.Model):
  3. first_name = models.CharField(max_length=30)
  4. last_name = models.CharField(max_length=40)
  5. email = models.EmailField()
  6. def __unicode__(self):
  7. return self.name
  8. class Meta:
  9. db_table = "author"
  10. class Book(models.Model):
  11. title = models.CharField(max_length=200)
  12. authors = models.ManyToManyField(Author)
  13. def __unicode__(self):
  14. return self.title
  15. class Meta:
  16. db_table = "book"

python manage.py sqlall books(App名) 看一下Django帮我们生成的数据库表结构:

  1. CREATE TABLE `author` (
  2. `id` integer AUTO_INCREMENT NOT NULL PRIMARY KEY,
  3. `first_name` varchar(30) NOT NULL,
  4. `last_name` varchar(40) NOT NULL,
  5. `email` varchar(75) NOT NULL
  6. )
  7. ;
  8. CREATE TABLE `book_authors` (
  9. `id` integer AUTO_INCREMENT NOT NULL PRIMARY KEY,
  10. `book_id` integer NOT NULL,
  11. `author_id` integer NOT NULL,
  12. UNIQUE (`book_id`, `author_id`)
  13. )
  14. ;
  15. ALTER TABLE `book_authors` ADD CONSTRAINT `author_id_refs_id_22051734` FOREIGN KEY (`author_id`) REFERENCES `author` (`id`);
  16. CREATE TABLE `book` (
  17. `id` integer AUTO_INCREMENT NOT NULL PRIMARY KEY,
  18. `title` varchar(200) NOT NULL
  19. )
  20. ;
  21. ALTER TABLE `book_authors` ADD CONSTRAINT `book_id_refs_id_77516490` FOREIGN KEY (`book_id`) REFERENCES `book` (`id`);

book表中并没有authors字段,其中的book_authors表就是用来存放书和作者多对多映射关系的。

访问多值:

一本书的所有作者:

b = Book.objects.get(id=50)

b.authors.all()

b.authors.filter(first_name='Adam')

反向也可以,一个作者的所有书:

a = Author.objects.get(id=1)

a.book_set.all()

给多对多字段添加值(添加多对多关系):

a = Author.objects.get(id=1)

b = Book.objects.get(id=50)

b.authors.add(a)

从多对多字段中删除值(删除多对多关系):

a = Author.objects.get(id=1)

b = Book.objects.get(id=50)

b.authors.remove(a) 或者 b.authors.filter(id=1).delete()

我想自定义关系表!!!

为什么?

可能的原因:

1)我想添加一些额外的字段;

2)与现有系统集成,关系表已经存在。

定义方法如下:

  1. # coding: utf-8
  2. from django.db import models
  3. class Author(models.Model):
  4. first_name = models.CharField(max_length=30)
  5. last_name = models.CharField(max_length=40)
  6. email = models.EmailField()
  7. def __unicode__(self):
  8. return self.name
  9. class Meta:
  10. db_table = "author"
  11. class Book(models.Model):
  12. title = models.CharField(max_length=200)
  13. authors = models.ManyToManyField(Author, through='BookAuthor') # 注意多了through参数
  14. def __unicode__(self):
  15. return self.title
  16. class Meta:
  17. db_table = "book"
  18. class BookAuthor(models.Model):
  19. book = models.ForeignKey(Book)
  20. author = models.ForeignKey(Author)
  21. created_at = models.DateTimeField(auto_now_add=True) # 我要添加额外的字段
  22. class Meta:
  23. db_table = "book_author_relationship" # 我需要用自定义的名称

再看一下此时的数据库表结构:

  1. BEGIN;
  2. CREATE TABLE `author` (
  3. `id` integer AUTO_INCREMENT NOT NULL PRIMARY KEY,
  4. `first_name` varchar(30) NOT NULL,
  5. `last_name` varchar(40) NOT NULL,
  6. `email` varchar(75) NOT NULL
  7. )
  8. ;
  9. CREATE TABLE `book` (
  10. `id` integer AUTO_INCREMENT NOT NULL PRIMARY KEY,
  11. `title` varchar(200) NOT NULL
  12. )
  13. ;
  14. CREATE TABLE `book_author_relationship` (
  15. `id` integer AUTO_INCREMENT NOT NULL PRIMARY KEY,
  16. `book_id` integer NOT NULL,
  17. `author_id` integer NOT NULL,
  18. `created_at` datetime NOT NULL
  19. )
  20. ;
  21. ALTER TABLE `book_author_relationship` ADD CONSTRAINT `author_id_refs_id_6b774382` FOREIGN KEY (`author_id`) REFERENCES `author` (`id`);
  22. ALTER TABLE `book_author_relationship` ADD CONSTRAINT `book_id_refs_id_7cf7763a` FOREIGN KEY (`book_id`) REFERENCES `book` (`id`);
  23. CREATE INDEX `book_author_relationship_752eb95b` ON `book_author_relationship` (`book_id`);
  24. CREATE INDEX `book_author_relationship_337b96ff` ON `book_author_relationship` (`author_id`);
  25. COMMIT;

爽!

但是这样做也是有不便的地方:多对多字段的add()和remove()方法就不能用了。这里有很好的解释:http://stackoverflow.com/questions/1755591/many-to-many-relationships-with-additional-data-on-the-relationship

https://docs.djangoproject.com/en/dev/topics/db/models/#extra-fields-on-many-to-many-relationships

那此时我们如何添加和删除映射关系呢?直接使用BookAuthor模型。

例如:

a = Author.objects.get(id=1)

b = Book.objects.get(id=50)

BookAuthor(book=b, author=a).save()

Django的Many-to-Many(多对多)模型的更多相关文章

  1. django一对多、多对多模型、自关联的建立

    # 原创,转载请留言联系 一对多模型 一对多的关系,例如员工跟部门.一个部门有多个员工.那么在django怎么建立这种表关系呢? 其实就是利用外键,在多的一方,字段指定外键即可.例如员工和部门,员工是 ...

  2. Django学习笔记之数据库-数据库与模型

    MySQL数据库 在网站开发中,数据库是网站的重要组成部分.只有提供数据库,数据才能够动态的展示,而不是在网页中显示一个静态的页面.数据库有很多,比如有SQL Server.Oracle.Postgr ...

  3. legend3---lavarel多对多模型操作实例

    legend3---lavarel多对多模型操作实例 一.总结 一句话总结: 在多对多模型中,增加关系表的数据 需要 弄一个和关系表一对多的模型关系 1.在lavarel关系模型中,课程和标签表是多对 ...

  4. Python Django框架笔记(五):模型

    #前言部分来自Django Book (一)    前言 大多数web应用本质上: 1. 每个页面都是将数据库的数据以HTML格式进行展现. 2. 向用户提供修改数据库数据的方法.(例如:注册.发表评 ...

  5. Django ORM 一对一,一对多,多对多, 添加,批量插入和查询

    模型类 class Book(models.Model): nid = models.AutoField(primary_key=True) title = models.CharField(max_ ...

  6. Django——7 常用的查询 常用的模型字段类型 Field的常用参数 表关系的实现

    Django 常用的查询 常用的查询方法 常用的查询条件 常用字段映射关系 Field常用参数 表关系的实现 查用的查询方法 这是需要用到的数据 from django.http import Htt ...

  7. django ajax MTV与MVC 多对多创建方式

    MTV与MVC MTV模型(django): M:模型层(models.py) T:templates V:views MVC模型: M:模型层(models.py) V:视图层(views.py) ...

  8. django 中models表的多对一,多对多的理解

    django 表的理解 好处:设计的好,会清晰,易于理解和维护,后期开发事半功倍,一目了然. 1. 一对一的表,两表的属性实际上完全可以合并成一个表,共用一个主键即可: 2. 一对多的表,可以设中间关 ...

  9. Django教程:[33]从数据库生成模型

    在使用django做网站的时候,有时候我们的数据库来自一个已有的数据库,如何整合这个数据库呢? django提供了方便的方法来整合已有数据库,下面我们看看具体的方法: 1.先来设置数据库:在网站文件夹 ...

  10. django “如何”系列8:如何为模型提供初始化数据

    当你第一次配置一个app的时候,有时候使用硬编码的数据去预填充你的数据库是非常有用的.这里有几个你可以让django自动创建这些数据的方法:你可以提供固定格式的初始化数据或者提供通过SQL初始化数据. ...

随机推荐

  1. 写sql语句分别按日,星期,月,季度,年统计

    --写sql语句分别按日,星期,月,季度,年统计销售额 --按日 ' group by day([date]) --按周quarter ' group by datename(week,[date]) ...

  2. Sql Server 检测死锁的SQL语句

    首先创建一个标量值函数DigLock,用来递归检测SqlServer中的每一个会话是否存在加锁循环,如果该函数最终返回1则表示检测到了加锁循环 (也就是说检测到了死锁),如果最终返回0则表示没有检测到 ...

  3. sql server 数据库模型 备份 恢复 总结 备份脚本

    事务日志是可以基于时间点恢复的,必须在full或bulk_logged模式下 Alter database [DBName] set recover bulk_logged , then the fo ...

  4. 精简高效的CSS命名准则/方法

    /* ---------------------single CSS----------------------- */ /* display */ .dn{display:none;} .di{di ...

  5. Linux的视频编程(V4L2编程)【转】

    本文转载自:http://blog.csdn.net/tommy_wxie/article/details/11472073 一.什么是video4linuxVideo4linux2(简称V4L2), ...

  6. 【python cookbook】【字符串与文本】2.在字符串的开头或结尾处做文本匹配

    问题:在字符串的开头或结尾处按照指定的文本模式做检查,例如检查文件的扩展名.URL协议类型等: 解决方法:使用str.startswith()和str.endswith()方法 >>> ...

  7. [转]Win7下安装配置sharepoint server 2010

    转自:http://blog.sina.com.cn/s/blog_5d93d7aa010151lp.html 要开发SharePoint 2010应用程序,开发人员必须构建一个SharePoint ...

  8. 关于PHP的正则表达式

    1.入门简介 简单的说,正则表达式是一种可以用于模式匹配和替换的强有力的工具.我们可以在几乎所有的基于UNIX系统的工具中找到正则表达式的身影,例如,vi编辑器,Perl或PHP脚本语言,以及awk或 ...

  9. [HTML]页面间传值的五种方法

    一.QueryString传值:1. 这是最简单的传值方式,但缺点是传的值会显示在浏览器的地址栏中且不能传递对象,只适用于传递简单的且安全性要求不高的整数值,例如: 2. 新建一个WEB项目,添加一个 ...

  10. Browser对象

    Window对象即浏览器中打开的窗口,当文档里面有框架(frame或者iframe标签)时,浏览器会为HTML文档创建一个window对象,并为每个框架创建一个额外的window对象. 属性close ...