假设一个用户想要创建一个由其他用户创建的项组成的集合。项的Mongoid文档具有版本控制,创建集的用户可能不喜欢项作者对集的项所做的更改。因此,我希望set文档引用项的特定版本,允许set作者根据需要更新项引用。我计划在set文档中添加一个item版本号数组,以及一些获取特定版本的set items和更新item版本的方法。你觉得这种方法合理吗?你将如何解决这个问题?

class Item
  include Mongoid::Document
  include Mongoid::Paranoia
  include Mongoid::Versioning
  field :title, type: String
  has_and_belongs_to_many :item_sets
end

class ItemSet
  include Mongoid::Document
  field :name, type: String
  field :item_versions, type: Array
  has_and_belongs_to_many :items
end

最佳答案

我通过在“中间”创建一个像“itemreference”这样的模型来解决这样的问题
MongoDB是文档存储,而不是关系数据库,因此在必要时存储重复信息是合法的。MongoDB有存储嵌入文档的能力,所以我们将使用这个伟大的特性。
itemreference保存有关创建视图所需的项的所有关键信息。这减少了视图端的查询,但增加了插入/更新端的查询。
问题是,您需要一个由item_id和版本号组成的“复合主键”。
让我们谈谈代码:
项目模型:

class Item
  include Mongoid::Document
  include Mongoid::Paranoia
  include Mongoid::Versioning

  field :title, :type => String

  # create a reference
  def to_reference
    # create new reference, containing all crucial attributes for displaying
    ItemReference.new(
      :item_id => self._parent.nil? ? self.id : self._parent.id,
      :version => self.version,
      :title => self.title
    )
  end

  # get a certain version of this item
  def get_version(version_number)
    return self if version_number == self.version
    self.versions.where(:version => version_number).first
  end

end

项目集模型
class ItemSet
  include Mongoid::Document
  field :name, :type => String

  embeds_many :item_references
end

项目引用模型
class ItemReference
  include Mongoid::Document
  embedded_in :item_sets

  field :title, :type => String

  # this points to the newest version
  belongs_to :item

  # get the original version
  def original
    self.item.get_version(self.version)
  end

  # update this reference to a certain version
  def update_to!(new_version)
    new_version     = self.item.get_version(new_version)
    if new_version.present?
      # copy attribute, except id
      self.attributes = new_version.to_reference.attributes.reject{|(k,v)| k == "_id"}
      self.save
    else
      # version not found
      false
    end
  end

  # update to the newest version
  def update_to_head!
    self.update_to!(self.item.version)
  end
end

此组合允许您创建包含具有不同版本的项的集合,并且可以将集合中的某些项引用更新为特定版本。
下面是一个例子:
first         = Item.create(:title => 'Item 1')
first.title   = 'Item 1.1'
first.save

myset         = ItemSet.create(:title => 'My Set')
myset.item_references << first.to_reference
myset.save

first.title   = 'Item 1.2'
first.save

p myset.item_references.first.title # prints Item 1.1
p myset.item_references.first.update_to_head!
p myset.item_references.first.title # prints Item 1.2
p myset.item_references.first.update_to!(1)
p myset.item_references.first.title # prints Item 1

08-27 21:16
查看更多