主キーの値を自動で設定してもらいたい

解決済

回答 1

投稿

  • 評価
  • クリップ 0
  • VIEW 4,124

shiho

score 12

MySQLを使っていて、たとえば以下のようなテーブルがあります。
CREATE TABLE body.artists
(
    artist_id                   INT NOT NULL,
    artist_name                 CHAR(200),
    description                 TEXT,
    CONSTRAINT PRIMARY KEY (artist_id)
);

このテーブルにレコードを追加するときに値を3カラム分指定しないといとエラーになります。
主キーのartist_idをinsertの時に指定するのではなく自動で連番にしてもらいたいのですが、どのように設定すれば良いのでしょうか?
  • 気になる質問をクリップする

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

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

    クリップを取り消します

  • 良い質問の評価を上げる

    以下のような質問は評価を上げましょう

    • 質問内容が明確
    • 自分も答えを知りたい
    • 質問者以外のユーザにも役立つ

    評価が高い質問は、TOPページの「注目」タブのフィードに表示されやすくなります。

    質問の評価を上げたことを取り消します

  • 評価を下げられる数の上限に達しました

    評価を下げることができません

    • 1日5回まで評価を下げられます
    • 1日に1ユーザに対して2回まで評価を下げられます

    質問の評価を下げる

    teratailでは下記のような質問を「具体的に困っていることがない質問」、「サイトポリシーに違反する質問」と定義し、推奨していません。

    • プログラミングに関係のない質問
    • やってほしいことだけを記載した丸投げの質問
    • 問題・課題が含まれていない質問
    • 意図的に内容が抹消された質問
    • 広告と受け取られるような投稿

    評価が下がると、TOPページの「アクティブ」「注目」タブのフィードに表示されにくくなります。

    質問の評価を下げたことを取り消します

    この機能は開放されていません

    評価を下げる条件を満たしてません

    評価を下げる理由を選択してください

    詳細な説明はこちら

    上記に当てはまらず、質問内容が明確になっていない質問には「情報の追加・修正依頼」機能からコメントをしてください。

    質問の評価を下げる機能の利用条件

    この機能を利用するためには、以下の事項を行う必要があります。

回答 1

checkベストアンサー

+2

オートインクリメントで実現できます。
https://dev.mysql.com/doc/refman/5.6/ja/example-auto-increment.html


 last_insert_id()関数について


last_insert_id()関数にFROM句は不要です。

公式のリファレンスによると、引数なしのlast_insert_id()の戻り値は
最近実行された INSERT ステートメントの結果として、最初に自動的に生成され、正常に AUTO_INCREMENT カラムに挿入された値を表す 64 ビット値
だからです。
https://dev.mysql.com/doc/refman/5.6/ja/information-functions.html#function_last-insert-id

なにやら難しいことを言っていますが、要するに
「最後に実行されたINSERT文で挿入されたレコードのうち、
  (レコードが複数ある場合は)最初に挿入されたレコードのオートインクリメント値」
が返却されるわけで、テーブルを指定することができないためです。
mysql> CREATE TABLE sample1 (id INT PRIMARY KEY AUTO_INCREMENT);
Query OK, 0 rows affected (0.01 sec)

# sample2テーブルはオートインクリメントを11から開始
mysql> CREATE TABLE sample2 (id INT PRIMARY KEY AUTO_INCREMENT) AUTO_INCREMENT = 11;
Query OK, 0 rows affected (0.01 sec)


mysql> INSERT INTO sample1 VALUES ();
Query OK, 1 row affected (0.01 sec)

mysql> INSERT INTO sample2 VALUES ();
Query OK, 1 row affected (0.00 sec)


# "FROM sample1"を付けても、返却されるのは最後にインサートしたsample2テーブルのオートインクリメント値
mysql> SELECT last_insert_id() FROM sample1;
+------------------+
| last_insert_id() |
+------------------+
|               11 |
+------------------+
1 row in set (0.00 sec)

mysql> SELECT last_insert_id();
+------------------+
| last_insert_id() |
+------------------+
|               11 |
+------------------+
1 row in set (0.00 sec)

加えて、FROM句を付けると、指定したテーブルに格納されているレコードの数だけ、結果セットの行が返ってしまいます。
mysql> INSERT INTO sample1 VALUES (), (), ();
Query OK, 3 rows affected (0.00 sec)
Records: 3  Duplicates: 0  Warnings: 0

mysql> SELECT last_insert_id() FROM sample1;
+------------------+
| last_insert_id() |
+------------------+
|                2 |
|                2 |
|                2 |
|                2 |
+------------------+
4 rows in set (0.00 sec)

mysql> SELECT last_insert_id();
+------------------+
| last_insert_id() |
+------------------+
|                2 |
+------------------+
1 row in set (0.00 sec)

投稿

編集

  • 回答の評価を上げる

    以下のような回答は評価を上げましょう

    • 正しい回答
    • わかりやすい回答
    • ためになる回答

    評価が高い回答ほどページの上位に表示されます。

  • 回答の評価を下げる

    下記のような回答は推奨されていません。

    • 間違っている回答
    • 質問の回答になっていない投稿
    • スパムや攻撃的な表現を用いた投稿

    評価を下げる際はその理由を明確に伝え、適切な回答に修正してもらいましょう。

  • 2015/10/31 19:59

    そして、自動発行された番号を取得できます!
    select last_insert_id() from body.artists;

    PHPでPDOお使いでしたら、簡単に取得できます~
    $id = $dbh->lastInsertId();

    この仕組みはパラレルに大量にクエリが発行されている状況でも、必ず自身のコネクションで実行された直前の値が取得できます。ご参考まで。

    キャンセル

  • 2015/10/31 20:42

    Ken.sakanakana様、コメントありがとうございます。

    last_insert_id()関数の使用方法について気になる点がございましたので、
    回答のほうに追記させていただきました。

    キャンセル

  • 2015/11/01 01:40

    ここまで詳しく書いて頂いてありがとうございます!これで少しでも同じ悩みの方が、この投稿をみて躓かないようになれば良いですね。

    キャンセル

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

  • ただいまの回答率 90.22%
  • 質問をまとめることで、思考を整理して素早く解決
  • テンプレート機能で、簡単に質問をまとめられる