質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
87.20%
Ruby on Rails

Ruby on Railsは、オープンソースのWebアプリケーションフレームワークです。「同じことを繰り返さない」というRailsの基本理念のもと、他のフレームワークより少ないコードで簡単に開発できるよう設計されています。

Active Record

Active Recordは、一つのオブジェクトに対しドメインのロジックとストレージの抽象性を結合するデザインパターンです。

解決済

ActiveRecordのpreloadとincludesの仕組みについて

dialbird
dialbird

総合スコア379

Ruby on Rails

Ruby on Railsは、オープンソースのWebアプリケーションフレームワークです。「同じことを繰り返さない」というRailsの基本理念のもと、他のフレームワークより少ないコードで簡単に開発できるよう設計されています。

Active Record

Active Recordは、一つのオブジェクトに対しドメインのロジックとストレージの抽象性を結合するデザインパターンです。

1回答

0評価

0クリップ

4426閲覧

投稿2017/02/22 09:00

失礼いたします。

ActiveRecordの挙動に困っています。

一応予備知識として、preloadではJoinしないためにwhereでの絞り込みができず、includesは状況に応じてeager_loadしてJoinしてくれるのでwhereの絞り込みができるということがわかっています。

ただこの間、preloadでもscope内でのjoinとmergeを使えば絞り込みができるということを聴きました。

そこで、一対多の関係にあるUserとDogモデルを用意して、

ruby

# User class User < ApplicationRecord has_many :dogs scope :dog_name_like, -> (name) { joins(:dogs).merge(Dog.name_like(name)) } end # Dog class Dog < ApplicationRecord belongs_to :user scope :name_like, -> (name) { where(arel_table[:name].matches("%#{name}%")) } end

以下のように、「犬の名前に"ab"の文字列が入っている飼い主」で検索してみたところ

ruby

# 以下のような呼び出しをする User.all.preload(:dogs).dog_name_like("ab")

確かにできはしました。
ただ、abの文字が入っている犬を多数持っている飼い主が多数いた場合、以下のSQLのように、なぜかまた犬をデータベースから呼び出すということをしていました。

SQL

/*userのロード*/ SELECT "users".* FROM "users" INNER JOIN "dogs" ON "dogs"."user_id" = "users"."id" WHERE ("dogs"."name" LIKE '%ab%') ORDER BY "users"."id" ASC /*dogのプリロード*/ SELECT "dogs".* FROM "dogs" WHERE "dogs"."user_id" IN (3, 4, 6, 8, 11, 12, 16, 17, 18, 19, 21, 23, 24, 26, 29, 36, 38, 40, 44, 45, 47, 49, 50, 51, 53, 57, 60, 62, 66, 69, 71, 74, 75, 77, 80, 92, 94, 95, 97, 100) /*??? 謎の呼び出し このuser_idは、紐づいているDogオブジェクトが複数存在するuserのid */ SELECT "dogs".* FROM "dogs" WHERE "dogs"."user_id" = ? [["user_id", 53]] SELECT "dogs".* FROM "dogs" WHERE "dogs"."user_id" = ? [["user_id", 60]] SELECT "dogs".* FROM "dogs" WHERE "dogs"."user_id" = ? [["user_id", 60]]

ちなみにpreloadの部分をincludesにすると

SELECT "users"."id" AS t0_r0, "users"."name" AS t0_r1, "users"."created_at" AS t0_r2, "users"."updated_at" AS t0_r3, "dogs"."id" AS t1_r0, "dogs"."name" AS t1_r1, "dogs"."user_id" AS t1_r2, "dogs"."created_at" AS t1_r3, "dogs"."updated_at" AS t1_r4 FROM "users" INNER JOIN "dogs" ON "dogs"."user_id" = "users"."id" WHERE ("dogs"."name" LIKE '%ab%') ORDER BY "users"."id" ASC

という理想のSQLになります。

そこで質問というのは、

  1. なぜpreloadだと読み込んで入るはずなのにもう一度DBに取りに行こうとするのか
  2. 結局joinとmergeの組み合わせはpreloadでやってはいけないのか。

ということです。

長くなりましたが、よろしくお願いいたします。

良い質問の評価を上げる

以下のような質問は評価を上げましょう

  • 質問内容が明確
  • 自分も答えを知りたい
  • 質問者以外のユーザにも役立つ

評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

気になる質問をクリップする

クリップした質問は、後からいつでもマイページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

  • プログラミングに関係のない質問
  • やってほしいことだけを記載した丸投げの質問
  • 問題・課題が含まれていない質問
  • 意図的に内容が抹消された質問
  • 過去に投稿した質問と同じ内容の質問
  • 広告と受け取られるような投稿

評価を下げると、トップページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

まだ回答がついていません

会員登録して回答してみよう

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
87.20%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問

同じタグがついた質問を見る

Ruby on Rails

Ruby on Railsは、オープンソースのWebアプリケーションフレームワークです。「同じことを繰り返さない」というRailsの基本理念のもと、他のフレームワークより少ないコードで簡単に開発できるよう設計されています。

Active Record

Active Recordは、一つのオブジェクトに対しドメインのロジックとストレージの抽象性を結合するデザインパターンです。