通过中间表,指定多对对关联。除非连接表显示指定为一个选项。因此developers与projects通过此关联,中间表将默认为developersprojects表,它是通过<运算符来计算的。如果需要,可以通过jointable来指定中间表的名称。
这个中间表不应该有主键或是模型关联。你必须向下面这样生成中间表:
class CreateDevelopersProjectsJoinTable < ActiveRecord::Migration def change create_table :developers_projects, :id => false do |t| t.integer :developer_id t.integer :project_id end end end
这也是一个好主意,添加索引,以加快这些列的加入过程。然而在MySql中建议添加两列的复合索引,mysql值使用一个索引表中查找。
添加了以下检索和查询方法:
collection(force_reload = false) 返回所有一个包含所有关联对象的数组,如果不存在,则返回空数组。
collection<<(object, ...) 将一个或多个对象添加到关联对象集合。注意,这个操作立即触发更新sql,而不会等到保存或更新。
collection.delete(object, ...) 从关联集合中删除一个或多个对象的关联。这并不会销毁对象。
collection=objects 替换关联集合,通过删除和添加对象。
collection_singular_ids 返回一个数组,包括管理对象的所有id。
collection_singular_ids=ids 替换集合对象中主键集合。
collection.clear 清空关联集合。这并不会销毁对象。
collection.empty? 如果没有关联对象,返回true。
collection.size 返回关联对象的数量。
collection.find(id) 从关联对象中查找。它等同于ActiveRecord::Base.find(id).
collection.exists? 检查是否有一个关联对象符合给定的条件。它等同于ActiveRecord::Base.exists?
collection.build(attributes = {}) 返回一个新的集合类型,已实例化属性通过连接表链接到这个对象,但尚未保存。
collection.create(attributes = {}) 返回以实例化的属性集合类型,通过连接表链接到该对象的新对象,并已保存(如果它通过了验证)。这个会创建关联对象,然后写入中间表。这样,关联集合就又多了一个元素。
(collection 会被替换为第一个符号参数,所以has_one_belong\s_to_many :categories将会添加categories.empty?方法)
例子:
一个Developer类定义hasandbelongstomany :projects, 将会添加:
Developer#projects
Developer#projects<<
Developer#projects.delete
Developer#projects=
Developer#project_ids
Developer#project_ids=
Developer#projects.clear
Developer#projects.empty?
Developer#projects.size
Developer#projects.find(id)
Developer#projects.exists?(...)
Developer#projects.build(similar to Project.new("developer_id" => id))
Developer#projects.create(similar to c = Project.new("developer_id" => id); c.save; c)
这个定义语句可以包含一个hash选项。
:class_name,指定关联的类名。仅当不能从关联名称中推断时使用。
:jointable 指定中间表的名称。注意: 如果你重写了表面,必须在hasand_belongstomany两端都指定中间表名称。
:foreignkey 指定关联使用的外键。默认它等于#{class}id.
:associationforeignkey 指定关联的另一端对应中间表的外键。默认他等于关联的类#{class}_id.
:conditions 指定关联对象必须满足的where条件
:order 指定关联对象返回时执行一段order by sql片段,例如 last_name desc
:uniq 如果为true,重复的的关联对象访问与查询方法将被忽略。
:finder_sql 覆盖默认生成的sql语句,用于获取与主动声明
:countersql 指定一个用于获取关联数量的sql语句,如果指定了:findersql但没有指定:countersql,:countersql将会生成select count(*) from语句
:delete_sql 重写删除关联的sql语句。
:insert_sql 重写用于添加关联的sql语句。
:extend 匿名模块扩展代理,请参阅"Association extensions"
:include 用于指定依赖对象的依赖,当当前对象加载依赖时,include指定的依赖的依赖对象也会自动加载。
:group 按一个属性名称分组。使用GROUP BY
:having 结合:group 筛选group by 返回的数据。使用HAVING子句。
:limit 指定应该返回数据的长度
:offset 返回数据,偏移值。
:select 默认情况下,它使用select * from,但是可以改变它,不要加入join 列。不要忘了主键和外键,否则它会抛出错误。
:readonly 如果为true,所有关联对象将是只读的。
:validate 如果为false,保存时不验证关联对象。默认为true.
:autosave 如果为true,总是保存关联对象或销毁它们。如果为false,不会保存或销毁关联对象。默认情况下,值保存新的关联对象。
参数例子:
has_and_belongs_to_many :projects has_and_belongs_to_many :projects, :include => [ :milestones, :manager ] has_and_belongs_to_many :nations, :class_name => "Country" has_and_belongs_to_many :categories, :join_table => "prods_cats" has_and_belongs_to_many :categories, :readonly => true has_and_belongs_to_many :active_projects, :join_table => 'developers_projects', :delete_sql => "DELETE FROM developers_projects WHERE active=1 AND developer_id = #{id} AND project_id = #{record.id}"