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

Railsのuniquenessバリデーション完全理解!初心者でも失敗しない正解パターン(スコープ・大文字小文字・DB制約)

uniquenessの正解:スコープ・case_sensitive・DBユニークインデックスの併用
uniquenessの正解:スコープ・case_sensitive・DBユニークインデックスの併用

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

生徒

「Railsで同じメールアドレスを登録できないようにしたいんですが、どうすればいいですか?」

先生

「Railsでは、モデルにuniquenessバリデーションを書くことで重複を防げます。」

生徒

「それを書けば完璧なんですか?ネットでDBも設定しないとダメって見ました…」

先生

「その通りです。uniquenessには正しい使い方があります。今日は“本当に安全な方法”を順番に説明します。」

1. uniquenessバリデーションとは?

1. uniquenessバリデーションとは?
1. uniquenessバリデーションとは?

Railsのuniquenessバリデーションは、「同じ値がデータベースに存在しないこと」を確認する仕組みです。 たとえば、ユーザー登録で同じメールアドレスが二回使われないようにする、といった場面で使われます。

プログラミング未経験の方は、「受付名簿で同じ名前を二回書かせないチェック係」だと考えると分かりやすいです。 Railsでは、このチェック係をモデルに書くことで、自動的に確認してくれます。

2. 基本的なuniquenessの書き方

2. 基本的なuniquenessの書き方
2. 基本的なuniquenessの書き方

まずは一番シンプルな書き方です。Userモデルでメールアドレスの重複を防ぐ例を見てみましょう。


class User < ApplicationRecord
  validates :email, uniqueness: true
end

この設定により、すでに存在するemailと同じ値を保存しようとすると、エラーになります。 ただし、この書き方には落とし穴があります。

3. scopeを使ったuniquenessの正しい考え方

3. scopeを使ったuniquenessの正しい考え方
3. scopeを使ったuniquenessの正しい考え方

scope(スコープ)とは、「どの範囲で重複をチェックするか」を指定するものです。 たとえば、「同じ会社の中だけで社員番号が重複しないようにしたい」場合があります。


class Employee < ApplicationRecord
  validates :employee_number, uniqueness: { scope: :company_id }
end

これは「company_idが同じ場合だけ、employee_numberは一意である」という意味です。 学校で例えると、「クラスが違えば出席番号が同じでもOK」という状態です。

scopeを使わないと、「全会社共通で重複不可」という意図しないルールになることがあるため、実務では非常によく使われます。

4. case_sensitiveで大文字・小文字の違いを理解する

4. case_sensitiveで大文字・小文字の違いを理解する
4. case_sensitiveで大文字・小文字の違いを理解する

case_sensitiveは、「大文字と小文字を区別するかどうか」を指定するオプションです。 メールアドレスでは、通常は区別しない方が安全です。


class User < ApplicationRecord
  validates :email, uniqueness: { case_sensitive: false }
end

この設定をすると、「test@example.com」と「TEST@example.com」は同じものとして扱われます。 区別してしまうと、見た目がほぼ同じアカウントが複数作れてしまい、トラブルの原因になります。

5. uniquenessだけでは危険な理由

5. uniquenessだけでは危険な理由
5. uniquenessだけでは危険な理由

ここが初心者の方が一番つまずきやすいポイントです。 uniquenessバリデーションは、アプリ側のチェックにすぎません。

同時に二人が同じデータを登録しようとした場合、チェックをすり抜けて保存される可能性があります。 これは「レジが一台しかなくて、同時に確認できない状態」に似ています。

そのため、本当に安全にするには、データベース側でも重複禁止を設定する必要があります。

6. DBユニークインデックスの設定方法

6. DBユニークインデックスの設定方法
6. DBユニークインデックスの設定方法

ユニークインデックスとは、「この列は重複してはいけない」というルールをDBに直接設定する仕組みです。 Railsではマイグレーションで設定します。


class AddUniqueIndexToUsersEmail < ActiveRecord::Migration[7.0]
  def change
    add_index :users, :email, unique: true
  end
end

これにより、Railsのチェックをすり抜けても、DBが最後の砦として重複保存を防ぎます。 実務では「バリデーション+DB制約」は必須の組み合わせです。

7. scope付きユニークインデックスの実例

7. scope付きユニークインデックスの実例
7. scope付きユニークインデックスの実例

scopeを使ったuniquenessの場合、DB側も同じ条件で制約をかける必要があります。


class AddUniqueIndexToEmployees < ActiveRecord::Migration[7.0]
  def change
    add_index :employees, [:company_id, :employee_number], unique: true
  end
end

これで「同じ会社内では社員番号が重複しない」ことが、RailsとDBの両方で保証されます。

8. before_*コールバックと組み合わせる考え方

8. before_*コールバックと組み合わせる考え方
8. before_*コールバックと組み合わせる考え方

before_validationなどのコールバックを使うと、バリデーション前に値を整えることができます。 たとえば、メールアドレスをすべて小文字に統一する場合です。


class User < ApplicationRecord
  before_validation :downcase_email
  validates :email, uniqueness: { case_sensitive: false }

  private

  def downcase_email
    self.email = email.downcase if email.present?
  end
end

これにより、入力の揺れを減らし、uniquenessが正しく機能するようになります。 初心者の方は「事前に形をそろえてからチェックする」と覚えると理解しやすいです。

カテゴリの一覧へ
新着記事
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で比較演算子を完全解説!==・===・<=>・eql? の使い分け
No.7
Java&Spring記事人気No7
Ruby
Rubyのselect/reject/filterの使い方を完全解説!初心者向けの条件抽出レシピ
No.8
Java&Spring記事人気No8
データベース
PostgreSQLで順位付け!ROW_NUMBER関数の使い方を初心者向けに徹底解説