zoukankan      html  css  js  c++  java
  • Django 一对多,多对多关系解析

    【转】Django 一对多,多对多关系解析  

     
    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类中说明,关联表会自动建立。
     
    访问方式: 
    以下代码摘自 StackOverflow 用来说明访问方式
     

    Model Code

    from django.db import models
    
    class Engine(models.Model):
        name = models.CharField(max_length=25)
    
        def __unicode__(self):
            return self.name
    
    class Car(models.Model):
        name = models.CharField(max_length=25)
        engine = models.OneToOneField(Engine)
    
        def __unicode__(self):
            return self.name
    
    class Engine2(models.Model):
        name = models.CharField(max_length=25)
    
        def __unicode__(self):
            return self.name
    
    class Car2(models.Model):
        name = models.CharField(max_length=25)
        engine = models.ForeignKey(Engine2, unique=True)
    
        def __unicode__(self):
            return self.name

    OneToOneField Example

    >>> from testapp.models import Car, Engine
    >>> c = Car.objects.get(name='Audi')
    >>> e = Engine.objects.get(name='Diesel')
    >>> e.car  #engine 的model 定义中并没有car,这个是自动生成的,用关联表的类名小写直接访问
    <Car: Audi>  # 注意返回内容的不同

    ForeignKey with unique=True Example

    >>> from testapp.models import Car2, Engine2
    >>> c2 = Car2.objects.get(name='Mazda')
    >>> e2 = Engine2.objects.get(name='Wankel')
    >>> e2.car2_set.all()   # 在未定义的model中用关联表类名小写加"_set"来访问,多对多也一样
    [<Car2: Mazda>]   #注意返回内容的不同,这里是一个QuerySet
     
    以下内容摘自DjangoBook,仅供个人查询使用:
     
     
    from django.db import models
    
    class Publisher(models.Model):
        name = models.CharField(max_length=30)
        address = models.CharField(max_length=50)
        city = models.CharField(max_length=60)
        state_province = models.CharField(max_length=30)
        country = models.CharField(max_length=50)
        website = models.URLField()
    
        def __unicode__(self):
            return self.name
    
    class Author(models.Model):
        first_name = models.CharField(max_length=30)
        last_name = models.CharField(max_length=40)
        email = models.EmailField()
    
        def __unicode__(self):
            return u'%s %s' % (self.first_name, self.last_name)
    
    class Book(models.Model):
        title = models.CharField(max_length=100)
        authors = models.ManyToManyField(Author)
        publisher = models.ForeignKey(Publisher)
        publication_date = models.DateField()
    
        def __unicode__(self):
            return self.title
    

    如我们在第5章的讲解,获取数据库对象的特定字段的值只需直接使用属性。 例如,要确定ID为50的书本的标题,我们这样做:

     
     
    >>> from mysite.books.models import Book
    >>> b = Book.objects.get(id=50)
    >>> b.title
    u'The Django Book'
    

    但是,在之前有一件我们没提及到的是表现为ForeignKey 或 ManyToManyField的关联对象字段,它们的作用稍有不同。

     
     

    访问外键(Foreign Key)值

     

    当你获取一个ForeignKey 字段时,你会得到相关的数据模型对象。 例如:

     
     
    >>> b = Book.objects.get(id=50)
    >>> b.publisher
    <Publisher: Apress Publishing>
    >>> b.publisher.website
    u'http://www.apress.com/'
    

    对于用`` ForeignKey`` 来定义的关系来说,在关系的另一端也能反向的追溯回来,只不过由于不对称性的关系而稍有不同。 通过一个`` publisher`` 对象,直接获取 books ,用 publisher.book_set.all() ,如下:

     
     
    >>> p = Publisher.objects.get(name='Apress Publishing')
    >>> p.book_set.all()
    [<Book: The Django Book>, <Book: Dive Into Python>, ...]
    

    实际上,book_set 只是一个 QuerySet(参考第5章的介绍),所以它可以像QuerySet一样,能实现数据过滤和分切,例如:

     
     
    >>> p = Publisher.objects.get(name='Apress Publishing')
    >>> p.book_set.filter(name__icontains='django')
    [<Book: The Django Book>, <Book: Pro Django>]
    

    属性名称book_set是由模型名称的小写(如book)加_set组成的。

     
     

    访问多对多值(Many-to-Many Values)

     

    多对多和外键工作方式相同,只不过我们处理的是QuerySet而不是模型实例。 例如,这里是如何查看书籍的作者:

     
     
    >>> b = Book.objects.get(id=50)
    >>> b.authors.all()
    [<Author: Adrian Holovaty>, <Author: Jacob Kaplan-Moss>]
    >>> b.authors.filter(first_name='Adrian')
    [<Author: Adrian Holovaty>]
    >>> b.authors.filter(first_name='Adam')
    []
    

    反向查询也可以。 要查看一个作者的所有书籍,使用author.book_set ,就如这样:

     
     
    >>> a = Author.objects.get(first_name='Adrian', last_name='Holovaty')
    >>> a.book_set.all()
    [<Book: The Django Book>, <Book: Adrian's Other Book>]
    

    这里,就像使用 ForeignKey字段一样,属性名book_set是在数据模型(model)名后追加_set

  • 相关阅读:
    Android UI性能优化详解
    Android开发中一些被冷落但却很有用的类和方法
    Android开发:Translucent System Bar 的最佳实践
    让动画不再僵硬:Facebook Rebound Android动画库介绍
    Android Touch事件传递机制
    转载爱哥自定义View系列--Canvas详解
    Android最佳实践指南
    Android网络请求心路历程
    转载爱哥自定义View系列--文字详解
    ORM是什么?
  • 原文地址:https://www.cnblogs.com/fengjian2016/p/5377448.html
Copyright © 2011-2022 走看看