Rails3.2 has_and_belongs_to_many关联

浏览:1026 发布日期:2016-05-14 13:47:26
    has_and_belongs_to_many

通过中间表,指定多对对关联。除非连接表显示指定为一个选项。因此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选项。

Options

  • :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}"