假设一个用户想要创建一个由其他用户创建的项组成的集合。项的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