Usted puede saber si una asociación se carga con loaded?
.
Lo que está sucediendo aquí, si entiendo su problema, es que está tratando de ejecutar un buscador en una Relación ActiveRecord ::. Navegando rápidamente por the code, no parece que intente ver si una colección está cargada antes de emitir la consulta. Sin embargo, toma un bloque que evitará múltiples consultas. Por ejemplo (los nombres de los modelos se han cambiado porque estoy utilizando un proyecto de ejemplo que he creado para otra pregunta):
c = Canteen.first
Canteen Load (0.2ms) SELECT "canteens".* FROM "canteens" LIMIT 1
=> #<Canteen id: 1, name: "Really good place", created_at: "2012-12-13 00:04:11", updated_at: "2012-12-13 00:04:11">
c.meals.loaded?
=> false
c.meals.find {|m| m.id == 3}
Meal Load (0.2ms) SELECT "meals".* FROM "meals" WHERE "meals"."canteen_id" = 1
=> #<Meal id: 3, canteen_id: 1, name: "Banana Pie", price: #<BigDecimal:7fcb6784fa78,'0.499E1',18(45)>, created_at: "2012-12-13 00:37:41", updated_at: "2012-12-13 00:37:41">
Ves en el último ejemplo que ActiveRecord emite la consulta para cargar los registros asociados. Esto se debe a que ActiveRecord llama al to_a
en la asociación, lo que obliga a que se cargue todo el conjunto y luego se filtra según las condiciones del bloque. Obviamente, esto no es ideal.
Intentemos de nuevo, ansiosos por cargar la asociación.
c = Canteen.includes(:meals).first
Canteen Load (0.2ms) SELECT "canteens".* FROM "canteens" LIMIT 1
Meal Load (0.2ms) SELECT "meals".* FROM "meals" WHERE "meals"."canteen_id" IN (1)
=> #<Canteen id: 1, name: "Really good place", created_at: "2012-12-13 00:04:11", updated_at: "2012-12-13 00:04:11">
c.meals.loaded?
=> true
c.meals.find {|m| m.id == 3}
=> #<Meal id: 3, canteen_id: 1, name: "Banana Pie", price: #<BigDecimal:7fcb68b596f0,'0.499E1',18(45)>, created_at: "2012-12-13 00:37:41", updated_at: "2012-12-13 00:37:41">
En el último ejemplo aquí, verá que la colección no se carga de nuevo. En cambio, el bloque se usa para filtrar los registros ya cargados.
Como se puede ver a continuación, incluso si se cargan los registros, ActiveRecord emitirá una consulta para agarrar el registro asociado:
c.meals.loaded?
=> true
c.meals.find(1)
Meal Load (0.1ms) SELECT "meals".* FROM "meals" WHERE "meals"."canteen_id" = 1 AND "meals"."id" = ? LIMIT 1 [["id", 1]]
=> #<Meal id: 1, canteen_id: 1, name: "Enchiladas", price: #<BigDecimal:7fcb6584ce88,'0.699E1',18(45)>, created_at: "2012-12-13 00:04:40", updated_at: "2012-12-13 00:04:40">
SELECT "meals".* FROM "meals" WHERE "meals"."canteen_id" = 1 AND "meals"."id" = 3
=> [#<Meal id: 3, canteen_id: 1, name: "Banana Pie", price: #<BigDecimal:7fcb68b808e0,'0.499E1',18(45)>, created_at: "2012-12-13 00:37:41", updated_at: "2012-12-13 00:37:41">]
no es encontrar (: todo) obsoleto? – coorasse