ROR多对多 数据冗余问题。

向大家请教个问题。
模型:文档和标签 ,他们是多对多关系。
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