草庐IT

python - Django 中的 OneToOne、ManyToMany 和 ForeignKey 字段有什么区别?

coder 2023-05-22 原文

我在处理 Django 模型中的关系时遇到了一些困难。

有人可以解释 OneToOne、ManyToMany 和 ForeignKey 之间的区别吗?

最佳答案

嗯,这里基本上有两个问题:

  • 一对一、多对多、外键关系(一般)有什么区别
  • 它们与 Django 有什么区别。

  • 通过简单的谷歌搜索可以很容易地回答这两个问题,但是由于我在 SO 上找不到这个问题的确切重复,我会继续回答。

    请注意,在 Django 中,关系应该只在关系的一侧定义。

    外键

    外键关系通常称为多对一关系。请注意,这种关系的反向是一对多(Django 提供了访问工具)。顾名思义,许多对象可能与一个有关。
    Person >--| Birthplace
       ^           ^
       |           |
      Many        One 
    

    在这个例子中,一个人可能只有一个出生地,但一个出生地可能与很多人有关。让我们看看 Django 中的这个例子。说这些是我们的模型:
    class Birthplace(models.Model):
        city = models.CharField(max_length=75)
        state = models.CharField(max_length=25)
    
        def __unicode__(self):
            return "".join(self.city, ", ", self.state)
    
    class Person(models.Model):
        name = models.CharField(max_length=50)
        birthplace = models.ForeignKey(Birthplace)
    
        def __unicode__(self):
            return self.name
    

    您可以看到 Birthplace 中没有定义任何关系。模型和 ForeignKey关系在 Person 中定义模型。假设我们创建了模型的以下实例(显然不是 Python 语法):
  • 出生地:德克萨斯州达拉斯
  • 出生地:纽约市,纽约
  • 人:约翰史密斯,出生地:(德克萨斯州达拉斯)
  • 人:Maria Lee,出生地:(德克萨斯州达拉斯)
  • 人:Daniel Lee,出生地:(纽约市,纽约)

  • 现在我们可以看到 Django 如何让我们使用这些关系(注意 ./manage.py shell 是你的 friend !):
    >> from somewhere.models import Birthplace, Person
    >> Person.objects.all()
    [<Person: John Smith>, <Person: Maria Lee>, <Person: Daniel Lee>]
    >> Birthplace.objects.all()
    [<Birthplace: Dallas, Texas>, <Birthplace: New York City, New York>]
    

    您可以看到我们创建的模型实例。现在让我们检查某人的出生地:
    >> person = Person.object.get(name="John Smith")
    >> person.birthplace
    <Birthplace: Dallas, Texas>
    >> person.birthplace.city
    Dallas
    

    假设您想查看所有出生地的人。正如我之前所说,Django 允许您访问反向关系。默认情况下,Django 在您的模型上创建一个管理器( RelatedManager )来处理这个问题,名为 <model>_set ,其中 <model>是小写的型号名称。
    >> place = Birthplace.objects.get(city="Dallas")
    >> place.person_set.all()
    [<Person: John Smith>, <Person: Maria Lee>]
    

    请注意,我们可以通过设置 related_name 来更改此经理的名称。我们模型关系中的关键字参数。因此,我们将更改 birthplace Person 中的字段模型为:
    birthplace = models.ForeignKey(Birthplace, related_name="people")
    

    现在,我们可以用一个漂亮的名字来访问反向关系:
    >> place.people.all()
    [<Person: John Smith>, <Person: Maria Lee>]
    

    一对一

    一对一关系与多对一关系非常相似,不同之处在于它将两个对象限制为具有唯一关系。一个例子是用户和配置文件(存储有关用户的信息)。没有两个用户共享相同的配置文件。
    User |--| Profile
      ^          ^
      |          |
     One        One
    

    让我们在 Django 中看看这个。我不会费心定义用户模型,因为 Django 为我们定义了它。但是请注意,Django 建议使用 django.contrib.auth.get_user_model()导入用户,这就是我们要做的。配置文件模型可以定义如下:
    class Profile(models.Model):
        user = models.OneToOneField(settings.AUTH_USER_MODEL) # Note that Django suggests getting the User from the settings for relationship definitions
        fruit = models.CharField(max_length=50, help_text="Favorite Fruit")
        facebook = models.CharField(max_length=100, help_text="Facebook Username")
    
        def __unicode__(self):
            return "".join(self.fruit, " ", self.facebook)
    

    我们所需要的只是一个具有配置文件的用户,可以在 shell 中对此进行测试:
  • 用户:johndt6
  • 个人资料:用户:johndt6、“Kiwi”、“blah_blah”

  • 现在您可以轻松地从 User 模型访问用户的个人资料:
    >> user = User.objects.all()[0]
    >> user.username
    johndt6
    >> user.profile
    <Profile: Kiwi blah_blah>
    >> user.profile.fruit
    Kiwi
    >> profile = Profile.objects.get(user=user)
    >> profile.user
    <User: johndt6>
    

    当然,您可以使用 related_name 自定义反向关系的名称。论据如上。

    多对多

    多对多关系可能有点棘手。首先让我说多对多字段是困惑的,应该尽可能避免。鉴于此,在很多情况下多对多关系是有意义的。

    两个模型之间的多对多关系定义了第一模型的零个、一个或多个对象可能与第二模型的零个、一个或多个对象相关。例如,让我们设想一家通过项目定义其工作流程的公司。一个项目可能与没有订单、只有一个订单或多个订单相关。订单可能与无项目、一个项目或多个项目相关。
    Order >--< Project
      ^           ^
      |           |
     Many        Many
    

    让我们这样定义我们的模型:
    class Order(models.Model):
        product = models.CharField(max_length=150)  # Note that in reality, this would probably be better served by a Product model
        customer = models.CharField(max_length=150)  # The same may be said for customers
    
        def __unicode__(self):
            return "".join(self.product, " for ", self.customer)
    
    class Project(models.Model):
        orders = models.ManyToManyField(Order)
    
        def __unicode__(self):
            return "".join("Project ", str(self.id))
    

    请注意,Django 会创建一个 RelatedManagerorders字段来访问多对多关系。

    让我们为我们的模型创建以下实例(在我不一致的语法中!):
  • 订单:“宇宙飞船”、“美国宇航局”
  • 订单:“潜艇”、“美国海军”
  • 订购:“赛车”、“NASCAR”
  • 项目:订单:[]
  • 项目:订单:[(订单:“宇宙飞船”,“NASA”)]
  • 项目:orders: [(Order: "Spaceship", "NASA"), (Or​​der: "Race car", "NASCAR")]

  • 我们可以按如下方式访问这些关系:
    >> Project.objects.all()
    [<Project: Project 0>, <Project: Project 1>, <Project: Project 2>]
    >> for proj in Project.objects.all():
    ..     print(proj)
    ..     proj.orders.all()  # Note that we must access the `orders`
    ..                        # field through its manager
    ..     print("")
    Project 0
    []
    
    Project 1
    [<Order: Spaceship for NASA>]
    
    Project 2
    [<Order: Spaceship for NASA>, <Order: Race car for NASCAR>]
    

    请注意,NASA 订单与 2 个项目相关,而美国海军订单与一个项目无关。另请注意,一个项目没有订单,一个项目有多个订单。

    我们也可以以与之前相同的方式反向访问关系:
    >> order = Order.objects.filter(customer="NASA")[0]
    >> order.project_set.all()
    [<Project: Project 0>, <Project: Project 2>]
    

    ASCII 基数指南

    在我的 ASCII 图表可能有点困惑的情况下,以下解释可能会有所帮助:
  • ><意思是“对很多”
  • |表示“对一”

  • 所以... A --| B意味着 A 的一个实例只能与 B 的一个实例相关。

    A --< B意味着 A 的实例可以与 B 的许多实例相关。
    A >--< B相当于……
    A --< B
    A >-- B
    

    因此,关系的每个“侧面”或方向都可以单独阅读。把它们挤在一起很方便。

    扩展这些关系之一可能更有意义:
                   +---- John Smith
                   |
     Dallas|-------+---- Jane Doe
                   |
                   +---- Joe Smoe
    

    资源

    Good explanation of db relationships由@MarcB 提供

    Wikipedia page on Cardinality

    Django 文档:

    models.ForeignKey

    models.OneToOneField

    models.ManyToManyField

    One-to-one Relationships

    Many-to-many Relationships

    关于python - Django 中的 OneToOne、ManyToMany 和 ForeignKey 字段有什么区别?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/25386119/

    有关python - Django 中的 OneToOne、ManyToMany 和 ForeignKey 字段有什么区别?的更多相关文章

    1. ruby - 如何从 ruby​​ 中的字符串运行任意对象方法? - 2

      总的来说,我对ruby​​还比较陌生,我正在为我正在创建的对象编写一些rspec测试用例。许多测试用例都非常基础,我只是想确保正确填充和返回值。我想知道是否有办法使用循环结构来执行此操作。不必为我要测试的每个方法都设置一个assertEquals。例如:describeitem,"TestingtheItem"doit"willhaveanullvaluetostart"doitem=Item.new#HereIcoulddotheitem.name.shouldbe_nil#thenIcoulddoitem.category.shouldbe_nilendend但我想要一些方法来使用

    2. ruby - 为什么我可以在 Ruby 中使用 Object#send 访问私有(private)/ protected 方法? - 2

      类classAprivatedeffooputs:fooendpublicdefbarputs:barendprivatedefzimputs:zimendprotecteddefdibputs:dibendendA的实例a=A.new测试a.foorescueputs:faila.barrescueputs:faila.zimrescueputs:faila.dibrescueputs:faila.gazrescueputs:fail测试输出failbarfailfailfail.发送测试[:foo,:bar,:zim,:dib,:gaz].each{|m|a.send(m)resc

    3. ruby - 其他文件中的 Rake 任务 - 2

      我试图在一个项目中使用rake,如果我把所有东西都放到Rakefile中,它会很大并且很难读取/找到东西,所以我试着将每个命名空间放在lib/rake中它自己的文件中,我添加了这个到我的rake文件的顶部:Dir['#{File.dirname(__FILE__)}/lib/rake/*.rake'].map{|f|requiref}它加载文件没问题,但没有任务。我现在只有一个.rake文件作为测试,名为“servers.rake”,它看起来像这样:namespace:serverdotask:testdoputs"test"endend所以当我运行rakeserver:testid时

    4. ruby-on-rails - Ruby net/ldap 模块中的内存泄漏 - 2

      作为我的Rails应用程序的一部分,我编写了一个小导入程序,它从我们的LDAP系统中吸取数据并将其塞入一个用户表中。不幸的是,与LDAP相关的代码在遍历我们的32K用户时泄漏了大量内存,我一直无法弄清楚如何解决这个问题。这个问题似乎在某种程度上与LDAP库有关,因为当我删除对LDAP内容的调用时,内存使用情况会很好地稳定下来。此外,不断增加的对象是Net::BER::BerIdentifiedString和Net::BER::BerIdentifiedArray,它们都是LDAP库的一部分。当我运行导入时,内存使用量最终达到超过1GB的峰值。如果问题存在,我需要找到一些方法来更正我的代

    5. python - 如何使用 Ruby 或 Python 创建一系列高音调和低音调的蜂鸣声? - 2

      关闭。这个问题是opinion-based.它目前不接受答案。想要改进这个问题?更新问题,以便editingthispost可以用事实和引用来回答它.关闭4年前。Improvethisquestion我想在固定时间创建一系列低音和高音调的哔哔声。例如:在150毫秒时发出高音调的蜂鸣声在151毫秒时发出低音调的蜂鸣声200毫秒时发出低音调的蜂鸣声250毫秒的高音调蜂鸣声有没有办法在Ruby或Python中做到这一点?我真的不在乎输出编码是什么(.wav、.mp3、.ogg等等),但我确实想创建一个输出文件。

    6. ruby-on-rails - Rails 3 中的多个路由文件 - 2

      Rails2.3可以选择随时使用RouteSet#add_configuration_file添加更多路由。是否可以在Rails3项目中做同样的事情? 最佳答案 在config/application.rb中:config.paths.config.routes在Rails3.2(也可能是Rails3.1)中,使用:config.paths["config/routes"] 关于ruby-on-rails-Rails3中的多个路由文件,我们在StackOverflow上找到一个类似的问题

    7. ruby-on-rails - Rails - 子类化模型的设计模式是什么? - 2

      我有一个模型:classItem项目有一个属性“商店”基于存储的值,我希望Item对象对特定方法具有不同的行为。Rails中是否有针对此的通用设计模式?如果方法中没有大的if-else语句,这是如何干净利落地完成的? 最佳答案 通常通过Single-TableInheritance. 关于ruby-on-rails-Rails-子类化模型的设计模式是什么?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.co

    8. ruby - 什么是填充的 Base64 编码字符串以及如何在 ruby​​ 中生成它们? - 2

      我正在使用的第三方API的文档状态:"[O]urAPIonlyacceptspaddedBase64encodedstrings."什么是“填充的Base64编码字符串”以及如何在Ruby中生成它们。下面的代码是我第一次尝试创建转换为Base64的JSON格式数据。xa=Base64.encode64(a.to_json) 最佳答案 他们说的padding其实就是Base64本身的一部分。它是末尾的“=”和“==”。Base64将3个字节的数据包编码为4个编码字符。所以如果你的输入数据有长度n和n%3=1=>"=="末尾用于填充n%

    9. ruby - 解析 RDFa、微数据等的最佳方式是什么,使用统一的模式/词汇(例如 schema.org)存储和显示信息 - 2

      我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i

    10. ruby-on-rails - Rails - 一个 View 中的多个模型 - 2

      我需要从一个View访问多个模型。以前,我的links_controller仅用于提供以不同方式排序的链接资源。现在我想包括一个部分(我假设)显示按分数排序的顶级用户(@users=User.all.sort_by(&:score))我知道我可以将此代码插入每个链接操作并从View访问它,但这似乎不是“ruby方式”,我将需要在不久的将来访问更多模型。这可能会变得很脏,是否有针对这种情况的任何技术?注意事项:我认为我的应用程序正朝着单一格式和动态页面内容的方向发展,本质上是一个典型的网络应用程序。我知道before_filter但考虑到我希望应用程序进入的方向,这似乎很麻烦。最终从任何

    随机推荐