gpt4 book ai didi

ruby-on-rails - Rails 多态多对多关联

转载 作者:行者123 更新时间:2023-12-03 15:37:39 24 4
gpt4 key购买 nike

我正在尝试设置一种通用的相关对象网络。假设我有 4 个模型。

  • 预订
  • 电影
  • 标记
  • 类别

我希望能够做到:

book = Book.find(1)
book.relations << Tag.find(2)
book.relations << Category.find(3)
book.relations #=> [Tag#2, Category#3]

movie = Movie.find(4)
movie.relations << book
movie.relations << Tag.find(5)
movie.relations #=> [Book#1, Tag#5]

基本上我希望能够获取任何模型类(或我允许的模型类)的任意 2 个对象并声明它们是相关的。

显然我不想创建一大堆连接表。这似乎不是一个有很多直通关联,也不是一个多态关联。

这是 Rails 可以通过关联声明支持的东西,还是我应该在这里滚动我自己的逻辑?

最佳答案

从早期开始,对多态性的支持有了显着改善。您应该能够在 Rails 2.3 中通过为所有模型使用单个连接表来实现这一点——一个关系模型。

class Relation
belongs_to :owner, :polymorphic => true
belongs_to :child_item, :polymorphic => true
end

class Book
has_many :pwned_relations, :as => :owner, :class_name => 'Relation'
has_many :pwning_relations, :as => :child_item, :class_name => 'Relation'

# and so on for each type of relation
has_many :pwned_movies, :through => :pwned_relations,
:source => :child_item, :source_type => 'Movie'
has_many :pwning_movies, :through => :pwning_relations,
:source => :owner, :source_type => 'Movie'
end

这种数据结构的一个缺点是,您必须为可能相等的配对创建两个不同的角色。如果我想看我的书的所有相关电影,我必须将这些集合添加在一起:

( pwned_movies + pwning_movies ).uniq

这个问题的一个常见例子是社交网络应用程序中的“ friend ”关系。Insoshi 使用的一种解决方案是在连接模型上注册一个 after_create 回调(在本例中为 Relation),这会创建反向关系。 after_destroy 回调同样是必要的,但是通过这种方式,以一些额外的数据库存储为代价,您可以确信您将在单个数据库查询中获得所有相关电影。

class Relation
after_create do
unless Relation.first :conditions =>
[ 'owner_id = ? and owner_type = ? and child_item_id = ? and child_item_type = ?', child_item_id, child_item_type, owner_id, owner_type ]
Relation.create :owner => child_item, :child_item => owner
end
end
end

关于ruby-on-rails - Rails 多态多对多关联,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/953521/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com