タイトル: リダイレクト(header)
SEOタイトル: PHP リダイレクト (header Location) 完全ガイド(301/302/303/307/308 / exit / 出力前必須)
| この記事の要点 |
|
基本構文
必ず exit; (または die;) を付けてください。 Location ヘッダーを送ってもブラウザがリダイレクトする前に、PHP は残りのコードを最後まで実行します。認証チェックの直後でリダイレクトしているつもりが、続けて機密処理を実行してしまう事故が頻発します。
// ❌ 危険: リダイレクト後に処理が継続する
if (!isLoggedIn()) {
header('Location: /login');
// exit; を忘れている
}
processDelete($_GET['id']); // → 未ログインでも削除が走る!
// ✅ 正しい: exit で確実に止める
if (!isLoggedIn()) {
header('Location: /login');
exit;
}
processDelete($_GET['id']);
HTTP ステータスの使い分け
| コード | 意味 | メソッド変換 | 用途 |
|---|---|---|---|
| 301 | Moved Permanently (恒久) | POST → GET に変わる | URL 変更、ドメイン引越し、HTTP → HTTPS |
| 302 | Found (一時) | POST → GET に変わる | 古い実装の既定。302 でなく 303 推奨 |
| 303 | See Other | POST → GET (明示的) | POST 完了 → GET 結果画面 (PRG パターン) |
| 307 | Temporary Redirect | メソッド維持 | POST のまま別 URL で受け直したい |
| 308 | Permanent Redirect | メソッド維持 | 恒久的な URL 変更 + メソッド保持 |
「Cannot modify header information」エラー
PHP 最大の頻出エラーの一つ:
Warning: Cannot modify header information - headers already sent by
(output started at /var/www/index.php:12) in /var/www/login.php on line 25
原因はheader() を呼ぶ前にすでに何かが出力されていること。次のいずれか:
の前に空白や HTMLがある- BOM 付き UTF-8 で保存されている (先頭 3 バイトの不可視 BOM)
echo/print/var_dumpをすでに実行している- include 先のファイル末尾に空行がある
- PHP の Warning / Notice がすでに出力されている
対処
を省略する (PHP 専用ファイルでは閉じタグ不要)
// 3. BOM なし UTF-8 で保存
// 4. include ファイルの末尾も同様に注意
// 緊急回避: 出力バッファリング
ob_start();
echo "ここで出力されても OK (バッファに溜まる)";
header('Location: /target');
ob_end_clean(); // バッファ破棄
exit;
絶対 URL vs 相対 URL
HTTP 1.1 仕様上、Location ヘッダーは絶対 URL が原則ですが、HTTP 1.1 以降は相対 URL も受け入れます。可搬性を優先するなら絶対 URL:
JavaScript リダイレクト
PHP 出力後にどうしてもリダイレクトしたい場合の最終手段:
セキュリティ: オープンリダイレクト脆弱性
ユーザー入力をそのまま Location に使うのは危険:
Laravel での書き方
// シンプルなリダイレクト
return redirect('/home');
// 名前付きルート
return redirect()->route('dashboard');
// 直前のページに戻る (フォームエラー時)
return back()->withInput()->withErrors($errors);
// 外部 URL
return redirect()->away('https://example.com');
// HTTP ステータス指定
return redirect('/login', 302);
return redirect()->route('home', [], 301);
// アクション直接呼び出し
return redirect()->action([UserController::class, 'index']);
// セッションへフラッシュメッセージ付与
return redirect('/dashboard')->with('status', '保存しました');
FAQ
Q: 301 と 302 はどう違う?
A: 301 は恒久。検索エンジンが古い URL を捨てて新 URL を採用、ブラウザもキャッシュする。一時的変更なら 302/303 を使う。SEO 引越しは必ず 301。
Q: POST フォーム送信後のリダイレクトはどれ?
A: 303 See Other が正解。302 だと再 POST の可能性がある (古いブラウザ)。PRG パターン (Post-Redirect-Get) で二重送信防止。
Q: クッキーをセットしつつリダイレクトしたい
A: setcookie() も出力前に呼ぶ。setcookie('key', $val); header('Location: /'); exit; の順で OK。