4

我有 4 个模型,A、B、C 和 D

class A < ActiveRecord::Base
  has_many :B
  has_many :C, :through => :B
end  

class B < ActiveRecord::Base
  belongs_to :A  
  has_many   :C
  has_many   :D, :through => :C
end  

class C < ActiveRecord::Base    
  belongs_to :B
end    

class D < ActiveRecord::Base    
  belongs_to :C
end    

我有一个非常幼稚的实现,非常明显......

<% A.B.each do |b| %>
  <%= b.number %>
  <% b.C.each do |c| %>
    <%= c.name %>
  <% end %>
<% end %>

获得 All C for A 的最佳方法是什么?获得All D for A的最佳方法是什么?

我想使用带有“created_at”值的 order_by 子句来获取所有“C”,而不是遍历 B。

可能是我缺少一些 ActiveRecord 魔法吗?

我很感激任何帮助。

4

1 回答 1

6

首先,您需要进行一些更改。

  1. class C需要一个协会来D

    class C < ActiveRecord::Base
      belongs_to :B
      has_one :D
    end
    
  2. 如果你想访问A' Ds ,你也需要指定它。

    class A < ActiveRecord::Base
      has_many :B
      has_many :C, :through => :B
      has_many :D, :through => :C
    end
    

现在,要访问所有的A' Cs:

-> a = A.where(:id => 1).includes(:C).first
  A Load (0.2ms)  SELECT "as".* FROM "as" WHERE "as"."id" = 1 LIMIT 1
  B Load (0.1ms)  SELECT "bs".* FROM "bs" WHERE "bs"."a_id" IN (1)
  C Load (0.1ms)  SELECT "cs".* FROM "cs" WHERE "cs"."b_id" IN (1, 2)
 => #<A id: 1, created_at: "2012-01-10 04:28:42", updated_at: "2012-01-10 04:28:42"> 
-> a.C
 => [#<C id: 1, b_id: 1, created_at: "2012-01-10 04:30:10", updated_at: "2012-01-10 04:30:10">, #<C id: 2, b_id: 1, created_at: "2012-01-10 04:30:11", updated_at: "2012-01-10 04:30:11">, #<C id: 3, b_id: 2, created_at: "2012-01-10 04:30:21", updated_at: "2012-01-10 04:30:21">, #<C id: 4, b_id: 2, created_at: "2012-01-10 04:30:21", updated_at: "2012-01-10 04:30:21">]

请注意,当您调用a.C. 这是因为 ActiveRecord 知道您将希望通过调用访问找到的 's A,并生成最少数量的查询。's也是如此:CincludeD

-> a = A.where(:id => 1).includes(:D).first
  A Load (0.1ms)  SELECT "as".* FROM "as" WHERE "as"."id" = 1 LIMIT 1
  B Load (0.1ms)  SELECT "bs".* FROM "bs" WHERE "bs"."a_id" IN (1)
  C Load (0.1ms)  SELECT "cs".* FROM "cs" WHERE "cs"."b_id" IN (1, 2)
  D Load (0.1ms)  SELECT "ds".* FROM "ds" WHERE "ds"."c_id" IN (1, 2, 3, 4)

假设您想要 all A' Ds 但想要C's 已订购:

A.where(:id => 1).includes(:C).order('cs.created_at DESC').includes(:D)

请注意,您还可以将其设置为关联的默认值:

:order选项规定了接收关联对象的顺序(在 SQLORDER BY子句使用的语法中)。

class Customer < ActiveRecord::Base
  has_many :orders, :order => "date_confirmed DESC"
end
于 2012-01-10T04:45:04.093 回答