向大家请教个问题。
模型:文档和标签 ,他们是多对多关系。
class Document < ActiveRecord::Base
has_and_belongs_to_many :tags,
:foreign_key => "document_id"
end
class Tag < ActiveRecord::Base
has_and_belongs_to_many :documents,
:foreign_key => "tag_id"
end
(中间表已经建好)
在保存文档的时候同时关联标签
@document = Document.new(params[:document])
tag_names = params[:tag_names].to_s.split(" ")
for tag_name in tag_names
@document.tags << Tag.new(:name => tag_name)
end
@document.save
这样在Doc保存的时候,标签中间表都会产生数据。
问题是,标签表中不需要重复数据。可每次都会把新数据insert进去,怎么才能保持关联去掉重复的呢?
[b]问题补充:[/b]
[code="ruby"]tag_names = params[:tag_names].to_s.split(" ")
for tag_name in tag_names
tag = Tag.find :first , :conditions => ["name = :name",{:name => tag_name}]
if tag.nil?
@document.tags << Tag.new(:name => tag_name)
else
@document.tags << tag
end
end
[/code]
最后我这么写的。不知道有没有更好的方法
倒,是这个,http://liuqiang.iteye.com/blog/216444
for tag_name in tag_names
@document.tags << Tag.new(:name => tag_name)
end
改成
@document.tags = tag_names.collect {|tag|
Tag.new(:name => tag)
}
很抱歉, 记的是这样. 没时间试.
用插件吧
http://liuqiang.iteye.com/admin/blogs/216444
ActiveRecord的Dynamic attribute based finder支持find_or_create_by_xxx的语法:
[code="ruby"]tag_names.each{|name| @document.tags << Tag.find_or_create_by_name(name)}[/code]
是逻辑的问题吧,你是检测不管tag存在不存在,都新建一个中间记录,其实该tag在存在的情况下还需要判断下这个tag是不是已经和该document关联了吧
if tag.nil?
@document.tags << Tag.new(:name => tag_name)
elsif !@document.tags.include?(tag)
@document.tags << tag
end