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

Railsのhas_many :throughとHABTMの違いを完全解説!中間テーブルに属性を持たせる設計を初心者向けにやさしく説明

has_many :through と HABTM の違い:中間テーブルに属性を持たせる設計
has_many :through と HABTM の違い:中間テーブルに属性を持たせる設計

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

生徒

「Railsで多対多って聞いたんですけど、has_many :through とHABTMって何が違うんですか?」

先生

「Railsでは、多対多の関係を作る方法が2つあります。違いを知ると、データベースの設計がとても楽になりますよ。」

生徒

「中間テーブルって言葉も出てきて、よく分からなくなりました…」

先生

「大丈夫です。まずは身近なたとえから、ゆっくり説明していきましょう。」

1. Railsとアソシエーションの基本

1. Railsとアソシエーションの基本
1. Railsとアソシエーションの基本

Railsは、Webアプリケーションを作るための便利な仕組みがたくさん用意されたフレームワークです。フレームワークとは、アプリ作りに必要な部品やルールが最初からそろっている道具箱のようなものです。

Railsでは、データベースの表と表のつながりをアソシエーションと呼びます。たとえば「ユーザー」と「投稿」のように、データ同士がどう関係しているかをRailsに教える仕組みです。

2. 多対多とは何かを超シンプルに理解する

2. 多対多とは何かを超シンプルに理解する
2. 多対多とは何かを超シンプルに理解する

多対多とは、「AもBも、お互いにたくさん関係する状態」を意味します。例として「生徒」と「授業」を考えてみましょう。一人の生徒は複数の授業を受けますし、ひとつの授業には複数の生徒がいます。

このような関係は、Railsではそのままでは表現できないため、間に中間テーブルを作ります。中間テーブルとは、2つの表をつなぐための連絡帳のような存在です。

3. HABTMとは?とにかくシンプルな多対多

3. HABTMとは?とにかくシンプルな多対多
3. HABTMとは?とにかくシンプルな多対多

HABTMは「has_and_belongs_to_many」の略です。名前は長いですが、考え方はとても単純です。中間テーブルはありますが、Railsのモデルとしては扱いません。


class Student < ApplicationRecord
  has_and_belongs_to_many :lessons
end

class Lesson < ApplicationRecord
  has_and_belongs_to_many :students
end

この方法は、「ただ結びついているだけ」で十分なときに向いています。中間テーブルには、学生IDと授業IDだけが入ります。

4. has_many :throughとは?中間テーブルが主役

4. has_many :throughとは?中間テーブルが主役
4. has_many :throughとは?中間テーブルが主役

has_many :throughは、HABTMより一歩進んだ方法です。中間テーブルを、Railsのモデルとしてしっかり扱います。


class Student < ApplicationRecord
  has_many :enrollments
  has_many :lessons, through: :enrollments
end

class Enrollment < ApplicationRecord
  belongs_to :student
  belongs_to :lesson
end

このように、中間テーブル用のモデルを作ることで、ただの連結以上の意味を持たせることができます。

5. 中間テーブルに属性を持たせる理由

5. 中間テーブルに属性を持たせる理由
5. 中間テーブルに属性を持たせる理由

has_many :throughが本領を発揮するのは、中間テーブルに属性を持たせたいときです。属性とは、データの内容のことです。

たとえば「いつ授業を受け始めたか」「成績はどうだったか」などは、生徒でも授業でもなく、その組み合わせに対する情報です。


class Enrollment < ApplicationRecord
  belongs_to :student
  belongs_to :lesson

  # 例:成績や受講日を保存できる
end

6. HABTMではできないこと

6. HABTMではできないこと
6. HABTMではできないこと

HABTMでは、中間テーブルをモデルとして使わないため、バリデーションや追加の処理を書けません。バリデーションとは、データが正しいかチェックする仕組みです。

「この授業は一度しか登録できない」「成績は必ず入れる」などのルールを作りたい場合、HABTMでは対応が難しくなります。

7. データベース設計の考え方を初心者向けに

7. データベース設計の考え方を初心者向けに
7. データベース設計の考え方を初心者向けに

パソコンを使ったことがない人でも、「ノートに表を書く」と考えると分かりやすいです。1冊のノートに無理やり全部書くのではなく、役割ごとにノートを分け、必要なら連絡用のノートを作る。それが中間テーブルです。

Railsのhas_many :throughは、その連絡用ノートにメモ欄を作れるイメージです。

8. has_many :throughとHABTMの使い分け

8. has_many :throughとHABTMの使い分け
8. has_many :throughとHABTMの使い分け

単純につながっているだけならHABTM、あとから情報を増やす可能性があるならhas_many :throughを選びます。実際のRails開発では、将来の変更に強いhas_many :throughが選ばれることが多いです。

Railsのアソシエーション設計は、アプリ全体の使いやすさに直結します。最初は難しく感じても、身近な例に置き換えて考えると、少しずつ理解できるようになります。

カテゴリの一覧へ
新着記事
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関数の使い方を初心者向けに徹底解説