6

在 Mongoid 中,将文档推送到embeds_many关系中会自动将文档持久保存到数据库中。通常,这很好,但是当我需要跟踪对嵌入文档的更改时会遇到问题。

假设您有两个模型:

class List
  include Mongoid::Document
  embeds_many :items
  field :title
end

class Item
  include Mongoid::Document
  embedded_in :list
  field :name
end

这发生在 .changes 属性上:

list = List.new(title: 'List title')
list.save  #list is now persisted
item = Item.new(name: 'Item name')
item.changes  #returns Hash with {'name' => [nil, 'Item name']}
list.items << item  #saves item to database under the hood
item.changes  #returns empty Hash, because item was autosaved with list

我可以item.previous_changes用来检查在将项目推入列表之前所做的更改,但在我的特定情况下,这会给我带来各种麻烦以保持事情的可控性。

我想要实现的是能够初始化Item文档,然后将其添加到list(通过<<push)而不立即持久化。

我知道 Mongoid 确实提供了在embeds_many不持久的情况下设置关系的选项(请参阅http://mongoid.org/en/mongoid/docs/relations.html#embeds_many):

list.items.build(name: 'Another item')

问题在于 Mongoid 为您创建了 Item 实例。在我的情况下,embeds_many关系中的文档可能是 Item 的子类(例如SpecialItem < Item),它不适用于build. 但是,如果有人知道解决此限制的方法,我也很乐意接受它作为答案。

4

2 回答 2

5

回答我自己的问题:通过将父文档分配给子文档来解决问题,而不是将子文档添加到子列表中。

继续上面的例子,你应该做

item.list = list  #no database query

代替

list.items << item  #automatic database insert

设置父子引用而不将任何内容自动保存到数据库中。

于 2013-03-21T11:36:41.460 回答
2

要使用您的示例跟进“构建子类”问题,您可以:

list.items.build({
  name: "Another Item"
}, SpecialItem)

指定您希望 Mongoid 为您构建的(子)类。

于 2013-11-06T03:30:55.157 回答