8

大家好,我的项目有一些有趣的要求。我需要一种has_one关系,它要么是一个类,要么是另一个类,但没有继承。如果这是唯一的方法,我可以摆脱继承,但是两个关联记录具有完全不同的数据并且根本不相关。

我需要弄清楚的是以下内容。

# 1. Foo never belongs to anything.
# 2. Foo MUST have one assigned sub-record for validity.
# 3. Foo can only have either Bar or Baz assigned.
# 4. Bar and Baz have only ONE common property, and aren't
#    related in either data or implementation.

class Foo < ActiveRecord::Base
  # Attributes: id, name, value
  has_one :assignment, :foreign_key => 'assigned_to', :readonly => true
          # Could really use an :object_type for has_one here...
end

class Bar < ActiveRecord::Base
  # Attributes: name,...
end

class Baz < ActiveRecord::Base
  # Attributes: name,...
end

WhereFoo有一个赋值,类型为Baror Baz; 它们只共享一个公共列,所以也许我可以从中创建一个父对象。但是,如果我让它们继承自一个通用对象(当它们包含的数据确实是橙子和苹果时)我必须为记录制作一个表吗?如果记录是抽象记录,但孩子们不是,我可以侥幸逃脱吗?

我想现在你可以看到我的困难了。我对 RoR 很陌生,但到目前为止我很喜欢它。我敢肯定有办法解决这个问题,但如果我无法弄清楚它是什么,我会被诅咒的。

4

2 回答 2

8

您正在尝试对不符合关系数据库范式的东西进行建模。SQL 中的所有引用都有一个来源和一个目标。

FWIW,多态关联也是一种反模式,因为它打破了这条规则。当文档说您必须放弃参照完整性约束才能使其工作时,这应该是一个线索,表明它是一个损坏的设计!

您需要 Foo 有两种has_one关系:一种与 Bar,另一种与 Baz。然后实现一些类逻辑,以确保在任何 Foo 实例中只填充一个引用。也就是说,在对 Bar 和 Baz 的引用中,一个必须有值,另一个必须为零,但这是您的代码需要检查和执行的内容。

于 2008-12-17T16:38:04.307 回答
0

也许做到这一点的一种方法是在 Foo 中为 Bar 和 Baz 创建 to has-one 关联。然后创建一个名为 assignment 和 assignment= 的方法,这是访问 Bar 和 Baz 的唯一方法。您可以在 get 方法中检查两个 has_ones 中的哪一个不是 nil 并返回那个。在赋值方法中,您可以检查传入的变量的类型,并将正确的 has-one 关系设置为该对象并将另一个设置为 nil。这应该涵盖你所有的基础,而不会太复杂。

于 2008-12-17T16:31:39.547 回答