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

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

新規登録して質問してみよう
ただいま回答率
85.48%
Laravel

LaravelとはTaylor Otwellによって開発された、オープンソースなPHPフレームワークです。Laravelはシンプルで表現的なシンタックスを持ち合わせており、ウェブアプリケーション開発の手助けをしてくれます。

Q&A

解決済

1回答

2911閲覧

親テーブルに送れない変数がある(Field ' ' doesn't have a default value)

riz

総合スコア30

Laravel

LaravelとはTaylor Otwellによって開発された、オープンソースなPHPフレームワークです。Laravelはシンプルで表現的なシンタックスを持ち合わせており、ウェブアプリケーション開発の手助けをしてくれます。

0グッド

0クリップ

投稿2019/03/14 17:35

編集2019/03/15 13:42

前提・実現したいこと

商品検索システムのようなものを作っており,

  • 店名(shops)
  • 支店名(branchs)

を一対多の関係で対応させようとしています。
ER図

発生している問題・エラーメッセージ

View→コントローラー→テーブルと入力値(店名と支店名)を移していくプログラムを作ったのですが,以下のように支店名が送れないエラーが出てしまいました。
私個人では解決ができなかっため,思い当たる原因などあればご教示頂きたいです。

SQLSTATE[HY000]: General error: 1364 Field 'branch_id' doesn't have a default value (SQL: insert into `shops` (`name`, `updated_at`, `created_at`) values (店名, 2019-03-15 02:04:15, 2019-03-15 02:04:15))

該当のソースコード

【index.blade.php】(View)

php

1<form action="/" method="POST" class="form-horizontal"> 2 <input type="text" name="shop" class="form-control" placeholder="店名" autocomplete="off"> 3 <input type="text" name="branch" class="form-control" placeholder="支店" autocomplete="off"> 4</form>

【web.php】(Routes)

php

1Route::post('/', 'MyController@store');

【MyController.php】(Controller)

php

1public function store(Request $request) 2 { 3 // branchsテーブルへの書き込み 4 $branch = new branch; 5 $branch_posted = $request->input('branch'); 6 branch::updateOrCreate(['name'=> $branch_posted], ['name'=> $branch_posted]); 7 8 // shopsテーブルへの書き込み 9 $branch_id_added = branch::where('name', $branch_posted)->get(); // branchsに追加したid 10 shop::updateOrCreate(['branch_id' => $branch_id_added, 'name' => $request->input('shop')]); 11 return redirect('/'); 12 }

【shop.php】(Model)

php

1class shop extends Model 2{ 3 protected $fillable = ['name']; 4 5 public function items() 6 { 7 return $this->hasMany('App\branch'); 8 } 9}

【branch.php】(Model)

php

1class branch extends Model 2{ 3 public function shops() 4 { 5 return $this->belongsTo('App\shop'); 6 } 7}

試したこと

  • shopsテーブルやshopモデルの削除→再作成を行ってみましたが,同じ結果でした。
  • StrictModeに関する情報を参考に,database.phpにて 'mysql' => ['prefix_indexes' => false] としてみましたが,
"SQLSTATE[23000]: Integrity constraint violation: 1452 Cannot add or update a child row: a foreign key constraint fails (`my_db`.`shops`, CONSTRAINT `shops_branch_id_foreign` FOREIGN KEY (`branch_id`) REFERENCES `branchs` (`id`) ON DELETE CASCADE ON UPDATE CASCADE) (SQL: insert into `shops` (`name`, `updated_at`, `created_at`) values (店名, 2019-03-15 02:22:53, 2019-03-15 02:22:53)) ◀"

というようなエラーが出てしまいました。

補足情報(FW/ツールのバージョンなど)

OS:Windows10
Apache 2.4.38
PHP Version 7.1.26
Laravel Framework 5.8.4


かなりとんちんかんなことをしていましたが,回答頂いた情報+コントローラの下記の書き換えで動かすことができました。
後学のために。

php

1$shop_posted = $request->input('shop'); 2$branch_posted = $request->input('branch'); 3 4$shop = new shop; 5shop::updateOrCreate(['name'=> $shop_posted], ['name'=> $shop_posted]); 6 7// branch_shop 8$shop_added = shop::where('name', $shop_posted)->first(); // branchesに追加したオブジェクト 9$shop_added->branches()->updateOrcreate(['name' => $branch_posted]);

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

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

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

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答1

0

ベストアンサー

店名データ(shops)1 に対して、支店名データ(branchs)を複数(多)存在させたい、のですよね?

であれば、

店名shopsに、支店名idbranch_idを持たせるのではなく、
支店名branchsに店名idshop_idを持たせるべきだと思います。

今のままでは、

仕様の関係性からは、「店名」にデータがないと「支店名」データは作れないのに、
テーブルの関係性からは、「店名」にデータを登録するには、支店名データ(branch_id)が必要(※)・・・という矛盾した状態です。

(※)が、最初に提示されたエラー

SQLSTATE[HY000]: General error: 1364 Field 'branch_id' doesn't have a default value (SQL: insert into `shops` (`name`, `updated_at`, `created_at`) values (店名, 2019-03-15 02:04:15, 2019-03-15 02:04:15))

に繋がるのかと思います。 エラーメッセージから見るに、shops.branch_idはnullが許容されていないのでしょう。

仮にですが、今のまま、なんとか「店名」「支店名」のデータを登録できたとしましょう。

例)

shop{ id:1 branch_id:1 name:お店A } branch{ id:1 name:Aの支店ひとつめ }

ここで次に、「お店A」に対して、2つ目の支店名を登録したい場合、どうなるでしょうか・・・

branch{ id:2 name:Aの支店ふたつめ }

お店Aのbranch_idを更新したら、

shop{ id:1 branch_id:2 name:お店A }

「支店ひとつめ」との関係がなくなってしまいます。

もし、店名データを増やしたら、

shop{ id:1 branch_id:2 name:お店A } shop{ id:2 branch_id:2 name:お店A }

テーブルの関係は、1:Nにはなりませんし、
データ上「支店ひとつめ」と「支店ふたつめ」が同じ「お店A」に属している、とは判断できない状態になってしまいます。

投稿2019/03/15 02:23

編集2019/03/15 02:26
mix-peach

総合スコア1910

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

riz

2019/03/15 13:35

丁寧なご説明ありがとうございました。 ご指摘の通り,テーブル内に支店名に店名idを持たせるようにしたところ,きちんとプログラムを動かすことができました! 多対一という言葉に惑わされて,データベース設計の基礎が抜けていました。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.48%

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

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

質問する

関連した質問