カテゴリ: Rails 更新日: 2026/02/05

RailsのN+1問題を完全理解!Bullet導入でクエリ最適化を自動検知する方法

Bullet導入で検知自動化:設定・通知・false positive の抑制テク
Bullet導入で検知自動化:設定・通知・false positive の抑制テク

先生と生徒の会話形式で理解しよう

生徒

「Railsでアプリを作ると、だんだん動きが遅くなることがあるって聞いたんですが本当ですか?」

先生

「ありますね。その原因のひとつが、N+1問題と呼ばれるデータベースへの無駄な問い合わせです」

生徒

「無駄な問い合わせって、自分で気づけるんですか?」

先生

「そこで役立つのがBulletというRails用の便利なツールです。設定すると自動で教えてくれますよ」

1. RailsとN+1問題の超基本

1. RailsとN+1問題の超基本
1. RailsとN+1問題の超基本

Railsは、Rubyで作られたWebアプリケーションフレームワークです。フレームワークとは、アプリ作りに必要な部品やルールがあらかじめ用意された土台のことです。Railsでは、データベースとのやりとりをActive Recordという仕組みで行います。

N+1問題とは、「1回で済むはずのデータ取得が、何回も繰り返されてしまう状態」のことです。たとえば、親データを1回取得し、その子データを人数分だけ毎回取りに行くような動きです。これが起きると、画面表示が遅くなり、サーバーにも負担がかかります。

2. N+1問題を身近な例で理解する

2. N+1問題を身近な例で理解する
2. N+1問題を身近な例で理解する

N+1問題を、図書館で本を探す場面に例えてみます。まず受付で「このクラスの生徒一覧」を1回聞きます。そのあと、生徒一人ひとりについて「この人が借りている本」を毎回カウンターに聞きに行くとします。これがN+1問題です。

Railsでも同じで、モデル同士の関連(アソシエーション)を使うと、知らないうちにこの状態が起きます。


users = User.all
users.each do |user|
  puts user.posts.count
end

このコードは一見シンプルですが、Userを1回取得したあと、ユーザーの数だけPostを取りに行くため、N+1問題が発生します。

3. includes・eager_loadでの基本対策

3. includes・eager_loadでの基本対策
3. includes・eager_loadでの基本対策

Railsには、N+1問題を防ぐための仕組みが用意されています。その代表例がincludeseager_loadです。どちらも「先にまとめてデータを取っておく」考え方です。


users = User.includes(:posts)
users.each do |user|
  puts user.posts.count
end

includesを使うと、UserとPostを最初にまとめて取得します。これにより、無駄な問い合わせが減ります。eager_loadも似ていますが、内部で使われるSQLが少し異なります。初心者のうちは「includesは便利」と覚えておけば十分です。

4. Bulletとは何か?初心者向け解説

4. Bulletとは何か?初心者向け解説
4. Bulletとは何か?初心者向け解説

Bulletは、RailsアプリでN+1問題や不要なincludesを自動で見つけてくれるツールです。人の目では気づきにくい部分も、Bulletがリアルタイムで教えてくれます。

Bulletは、開発中だけ使うgem(便利な部品集)として導入します。gemとは、Rubyで使える追加機能のことです。


group :development do
  gem 'bullet'
end

5. Bulletの設定方法と通知の見方

5. Bulletの設定方法と通知の見方
5. Bulletの設定方法と通知の見方

Bulletを入れたら、設定ファイルに「どうやって通知するか」を書きます。通知とは、問題が見つかったときに教えてくれる方法のことです。


config.after_initialize do
  Bullet.enable = true
  Bullet.alert = true
  Bullet.rails_logger = true
end

alertは画面にポップアップ表示、rails_loggerはログに出力する設定です。これにより、ブラウザを見ているだけで「ここでN+1が起きています」と分かります。

6. false positiveを抑える考え方

6. false positiveを抑える考え方
6. false positiveを抑える考え方

false positiveとは、「実際には問題ないのに、問題があると誤って検知されること」です。Bulletは便利ですが、すべてが本当の問題とは限りません。

たとえば、将来使う予定でincludesを書いている場合でも、Bulletは「不要」と判断することがあります。その場合は、Bulletの警告内容をよく読み、「本当に直すべきか」を考えます。


Bullet.add_whitelist(
  type: :unused_eager_loading,
  class_name: 'User',
  association: :posts
)

この設定で、特定の警告を無視できます。初心者のうちは、無理に全部消そうとせず、内容を理解することが大切です。

7. Bulletを使うと何が嬉しいのか

7. Bulletを使うと何が嬉しいのか
7. Bulletを使うと何が嬉しいのか

Bulletを使う最大のメリットは、「自分では気づけない無駄」を教えてくれる点です。Rails初心者は、正しい書き方をしているつもりでも、裏側で大量のSQLが動いていることがあります。

Bulletは、先生が横で画面を指さして「ここ、ちょっと無駄だよ」と教えてくれるような存在です。includesやeager_loadと一緒に使うことで、Railsのクエリ最適化とN+1対策を自然に学べます。

カテゴリの一覧へ
新着記事
New1
Ruby
“すべてはオブジェクト”を体感!初心者向けRubyのオブジェクト指向入門【irbで学ぶ】
New2
Ruby
Rubyの標準入出力を完全ガイド!puts・print・pの違いとデバッグ活用法
New3
Ruby
Gemとは?RubyGemsとBundlerを初心者向けに完全解説!依存関係管理も図解でわかりやすく理解
New4
Ruby
Rubyの文字エンコーディング入門!UTF-8・マジックコメント・外部/内部エンコーディングを完全解説
人気記事
No.1
Java&Spring記事人気No1
Ruby
Rubyのreduceとinject入門!合計計算や集計を初心者向けに分かりやすく解説
No.2
Java&Spring記事人気No2
Ruby
Rubyの文字列エンコーディング完全ガイド!Encoding・force_encoding・encodeを初心者向け解説
No.3
Java&Spring記事人気No3
Ruby
Rubyの始め方ガイド:インストールから最初のHello Worldまで(Windows/Mac/Linux)
No.4
Java&Spring記事人気No4
データベース
PostgreSQLのWHERE句を徹底解説!初心者でもわかるSQLデータ抽出の基本
No.5
Java&Spring記事人気No5
Ruby
Rubyのfind/detect/find_indexを徹底解説!目的のデータを素早く探す方法
No.6
Java&Spring記事人気No6
Ruby
Rubyのselect/reject/filterの使い方を完全解説!初心者向けの条件抽出レシピ
No.7
Java&Spring記事人気No7
Ruby
Rubyで比較演算子を完全解説!==・===・<=>・eql? の使い分け
No.8
Java&Spring記事人気No8
データベース
PostgreSQLで順位付け!ROW_NUMBER関数の使い方を初心者向けに徹底解説