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

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

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

MySQL(マイエスキューエル)は、TCX DataKonsultAB社などが開発するRDBMS(リレーショナルデータベースの管理システム)です。世界で最も人気の高いシステムで、オープンソースで開発されています。MySQLデータベースサーバは、高速性と信頼性があり、Linux、UNIX、Windowsなどの複数のプラットフォームで動作することができます。

SQL

SQL(Structured Query Language)は、リレーショナルデータベース管理システム (RDBMS)のデータベース言語です。大きく分けて、データ定義言語(DDL)、データ操作言語(DML)、データ制御言語(DCL)の3つで構成されており、プログラム上でSQL文を生成して、RDBMSに命令を出し、RDBに必要なデータを格納できます。また、格納したデータを引き出すことも可能です。

Q&A

解決済

1回答

779閲覧

MySQLで外部キー制約を内部のテーブルに使う場合、どのように書くのか知りたいです

marorin

総合スコア12

MySQL

MySQL(マイエスキューエル)は、TCX DataKonsultAB社などが開発するRDBMS(リレーショナルデータベースの管理システム)です。世界で最も人気の高いシステムで、オープンソースで開発されています。MySQLデータベースサーバは、高速性と信頼性があり、Linux、UNIX、Windowsなどの複数のプラットフォームで動作することができます。

SQL

SQL(Structured Query Language)は、リレーショナルデータベース管理システム (RDBMS)のデータベース言語です。大きく分けて、データ定義言語(DDL)、データ操作言語(DML)、データ制御言語(DCL)の3つで構成されており、プログラム上でSQL文を生成して、RDBMSに命令を出し、RDBに必要なデータを格納できます。また、格納したデータを引き出すことも可能です。

0グッド

0クリップ

投稿2020/05/03 11:54

編集2020/05/03 12:16

###疑問点
以下はツイッターと同じ仕組みを想定し、parent_tweet_idは返信先のtweet_idを示すカラムになります。
以下でエラーにならないのが疑問です。

###疑問が生じた考え
以下には外部キー制約として

SQL

1FOREIGN KEY (`parent_tweet_id`) REFERENCES tweets(`tweet_id`)

とありますがこれは

tweetsテーブルを探して、parent_tweet_idと同じ値がtweetsテーブルのtweet_idになければいけません

という意味の制約だと理解しています。

しかしもしこの理解があっている場合、parent_tweet_idとしてINSERTされている999は、tweetsテーブルのtweet_idに存在しないのに、なぜエラーにならないのでしょうか?

またこのように、外部キー制約を内部のテーブルに使う場合、どのように書くのが一般的な処理方法なのでしょうか?

SQL

1CREATE TABLE tweets( 2 `tweet_id` int not null, 3 `user_id` int not null, 4 `parent_tweet_id` int, 5 `content` varchar(1000) not null, 6 INDEX idx_parent_tweet_id(parent_tweet_id), 7 PRIMARY KEY (`tweet_id`), 8 FOREIGN KEY (`parent_tweet_id`) REFERENCES tweets(`tweet_id`) 9); 10INSERT INTO tweets 11 (`tweet_id`, `user_id`, `parent_tweet_id`, `content`) 12VALUES 13 (1, 1, 0, '元気ですか?'), 14 (2, 2, 999, '元気です') #999は存在しないのにエラーにならない? 15;

###考えた処理方法
一般的な処理方法がわからず自分なりに思いついたのは…とても冗長に思えるのですが次の構造です。
つまり返信関係を示すためだけにparent_relationsというテーブルを別に用意するということですが、これはおかしいでしょうか?

SQL

1CREATE TABLE tweets( 2 `tweet_id` int not null, 3 `user_id` int not null, 4 `content` varchar(1000) not null, 5 PRIMARY KEY (`tweet_id`) 6); 7INSERT INTO tweets 8 (`tweet_id`, `user_id`, `content`) 9VALUES 10 (1, 1, '元気ですか?'), 11 (2, 2, '元気です') 12; 13 14CREATE TABLE parent_relations( 15 `parent_tweet_id` int not null, 16 `tweet_id` int not null, 17 PRIMARY KEY (`parent_tweet_id`), 18 FOREIGN KEY (`parent_tweet_id`) REFERENCES tweets(`tweet_id`), 19 FOREIGN KEY (`tweet_id`) REFERENCES tweets(`tweet_id`) 20); 21INSERT INTO parent_relations 22 (`parent_tweet_id`, `tweet_id`) 23VALUES 24 (999, 2) 25;

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

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

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

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

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

m.ts10806

2020/05/03 12:13

最後のINSERTが文法的におかしいです。調整願います。
marorin

2020/05/03 12:14

バックティック忘れですか?ありがとうございます。直しました。
hoshi-takanori

2020/05/03 15:50

外部キー制約が有効であれば、最初に parent_tweet_id に 0 を指定した時点でエラーになるはずです。 外部キー制約をサポートするストレージエンジン (InnoDB) をお使いでしょうか? また、foreign_key_checks の値は 1 になってますか?
marorin

2020/05/03 16:08

>外部キー制約をサポートするストレージエンジン (InnoDB) をお使いでしょうか? どうもありがとうございます。ENGINE=innoDB; とすることでエラーが出ました。 恐れながら引き続き親子関係の構造(tweetsの1つにするか、tweetsとparent_relationsの2つにするか)につきましてもご意見頂戴できましたら幸いです。
hoshi-takanori

2020/05/03 16:11

データベース設計はあまり詳しくありませんが、個人的には最初のやり方 (同じテーブルへの外部キー制約)で充分だと思います。
marorin

2020/05/03 16:22

やはり二番目のやり方は冗長に過ぎますよね。貴重なご意見をどうもありがとうございます。
guest

回答1

0

ベストアンサー

何れにしても、親子関係を示す状態のナイーブツリーですので、テーブルは分けない方がいいでしょう。
テーブル構造より制約を優先させるのは本末転倒です。制約はロジックで如何様にもなりますし。

それよりも、アンチパターンであるナイーブツリーのを許容するかどうかです。
再帰を採用するならありですけど、そうでないなら経路列挙の構造ですね。

以下参考
SQL: ナイーブツリーと経路列挙モデル
※上記は経路が可変なのでまだアンチパターンですが、上限を決めて固定のカラムにすることがアンチパターンでない方法です。
MySQL 8.0 Lab版: MySQLの (再帰)共通テーブル式(CTE)

経路列挙モデルについて、他のDBMSでは配列を使う方法もあります。

投稿2020/05/04 04:34

sazi

総合スコア25195

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

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

marorin

2020/05/04 05:13

>テーブル構造より制約を優先させるのは本末転倒です。 優先は制約でなく構造でしたか。どうもありがとうございます。そういったことを知りませんので大変助かります。 >それよりも、アンチパターンであるナイーブツリーのを許容するかどうかです。 こちらも知りませんでした。たしかに取得の際はjoinが増え続けてしまいますし、削除の際は関連性の変更が大きなコストですね。教えて頂き感謝です。 >以下参考 頂戴した参考リンクですと私には難しすぎまして、こちらのリンク(https://qiita.com/hirashunshun/items/06adf4f42f03a9f3b63d)でやっと理解できました。 様々な解決策がありつつ一長一短でもあり、実際に私のケースに使うべきは何が適切な戸惑いますね。 しかしこちらのリンクには「再帰」や「配列を使う」という解決策が見当たりませんでした。そこで下記質問させてくださいませ。 >再帰を採用するならあり 最後に再帰について教えて頂けませんでしょうか。再帰という処理はなんとなくわかるのですが、それをどう採用すると今回のナイーブツリーはありと考えられるのでしょうか。
marorin

2020/05/04 05:40 編集

でもよく考えたらTwitterは以下のように1階層のみ表示するようになっていて、 コメント1 ├ 1への返信2 └ 1への返信3 逆に以下のように多階層での表示はしませんから、 コメント1 ├ 1への返信2 └ 1への返信3 --└ 3への返信4 こう考えてみますと取得の際にjoinが増え続けるということはない(返信の階層は無際限だが、取得の際は1階層しか取得しない)ので、ナイーブツリーの採用でも問題なさそうに思いました。削除の際も問題ないでしょうし。 と、以上の考え方はあっていますでしょうか。
sazi

2020/05/04 10:01

親と子だけに限定するなら、階層数が可変とはならないので問題は無いでしょう。
marorin

2020/05/04 10:05

コメントありがとうございます。安心できました。
sazi

2020/05/04 10:10

ナイーブツリーの階層のように要素が可変になるものを扱うにはチューリング完全である必要がありますが、 CTE(再帰はこの一部)とwindow機能が実装されるまでSQLはチューリング完全ではありませんでした。 「再帰を採用するならあり」としたのは、再帰を採用しないとSQLのみでは可変に対応できないからです。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

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

ただいまの回答率
85.48%

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

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

質問する

関連した質問