web-dev-qa-db-ja.com

PostgreSQLの自動インクリメント

私はMySQLからPostgreSQLに切り替えていますが、オートインクリメント値をどのように設定できるか疑問に思いました。 PostgreSQLのドキュメントで "serial"というデータ型を見ましたが、それを使うと構文エラーになります(v8.0)。

527
Ian

はい、SERIALは同等の機能です。

CREATE TABLE foo (
id SERIAL,
bar varchar);

INSERT INTO foo (bar) values ('blah');
INSERT INTO foo (bar) values ('blah');

SELECT * FROM foo;

1,blah
2,blah

SERIALは、シーケンスを囲む単なるテーブルタイムマクロです。 SERIALを既存の列に変更することはできません。

657
Trey

smallintなど、他の 整数データ型 を使用できます。

例:

CREATE SEQUENCE user_id_seq;
CREATE TABLE user (
    user_id smallint NOT NULL DEFAULT nextval('user_id_seq')
);
ALTER SEQUENCE user_id_seq OWNED BY user.user_id;

ユーザー シリアルデータ型 ではなく、独自のデータ型を使用することをお勧めします。

214
Ahmad

既に存在するテーブルのidにシーケンスを追加したい場合は、次のようにします。

CREATE SEQUENCE user_id_seq;
ALTER TABLE user ALTER user_id SET DEFAULT NEXTVAL('user_id_seq');
93
sereja

シーケンスはMySQLのauto_incrementの に相当する のように見えますが、微妙ではあるが重要な違いがいくつかあります。

1.失敗したクエリはシーケンス/シリアルを増加させます

シリアル列は、失敗したクエリで増加します。これは、行の削除だけでなく、失敗したクエリからの断片化につながります。たとえば、PostgreSQLデータベースで次のクエリを実行します。

CREATE TABLE table1 (
  uid serial NOT NULL PRIMARY KEY,
  col_b integer NOT NULL,
  CHECK (col_b>=0)
);

INSERT INTO table1 (col_b) VALUES(1);
INSERT INTO table1 (col_b) VALUES(-1);
INSERT INTO table1 (col_b) VALUES(2);

SELECT * FROM table1;

次のような出力が得られるはずです。

 uid | col_b 
-----+-------
   1 |     1
   3 |     2
(2 rows)

Uidが1から2ではなく1から3に変わることに注目してください。

あなたが手動であなた自身のシーケンスを作成しようとしたならば、これはまだ起こります:

CREATE SEQUENCE table1_seq;
CREATE TABLE table1 (
    col_a smallint NOT NULL DEFAULT nextval('table1_seq'),
    col_b integer NOT NULL,
    CHECK (col_b>=0)
);
ALTER SEQUENCE table1_seq OWNED BY table1.col_a;

MySQLの違いをテストしたい場合は、MySQLデータベースで以下を実行してください。

CREATE TABLE table1 (
  uid int unsigned NOT NULL AUTO_INCREMENT PRIMARY KEY,
  col_b int unsigned NOT NULL
);

INSERT INTO table1 (col_b) VALUES(1);
INSERT INTO table1 (col_b) VALUES(-1);
INSERT INTO table1 (col_b) VALUES(2);

断片化なし で次のようになります。

+-----+-------+
| uid | col_b |
+-----+-------+
|   1 |     1 |
|   2 |     2 |
+-----+-------+
2 rows in set (0.00 sec)

2.手動でシリアル列の値を設定すると、将来のクエリが失敗する可能性があります。

これは前回の回答で@trevによって指摘されました。

これをシミュレートするには、手動でuidを4に設定します。これは後で "衝突"します。

INSERT INTO table1 (uid, col_b) VALUES(5, 5);

テーブルデータ:

 uid | col_b 
-----+-------
   1 |     1
   3 |     2
   5 |     5
(3 rows)

別の挿入を実行します。

INSERT INTO table1 (col_b) VALUES(6);

テーブルデータ:

 uid | col_b 
-----+-------
   1 |     1
   3 |     2
   5 |     5
   4 |     6

別の挿入を実行したとします。

INSERT INTO table1 (col_b) VALUES(7);

次のエラーメッセージが表示されて失敗します。

エラー:重複したキー値が一意の制約 "table1_pkey"に違反しています詳細:キー(uid)=(5)は既に存在します。

対照的に、MySQLは以下に示すようにこれを適切に処理します。

INSERT INTO table1 (uid, col_b) VALUES(4, 4);

今すぐuidを設定せずに別の行を挿入します

INSERT INTO table1 (col_b) VALUES(3);

クエリは失敗せず、5にジャンプするだけです。

+-----+-------+
| uid | col_b |
+-----+-------+
|   1 |     1 |
|   2 |     2 |
|   4 |     4 |
|   5 |     3 |
+-----+-------+

テストはLinux(x86_64)およびPostgreSQL 9.4.9用のMySQL 5.6.33で行われました。

39
Programster

Postgres 10以降では、標準SQLで定義されているidentityカラムもサポートされています。

create table foo 
(
  id integer generated always as identity
);

明示的に要求されない限り、オーバーライドできないID列を作成します。次の挿入はgenerated alwaysとして定義された列で失敗します。

insert into foo (id) 
values (1);

しかしこれは却下することができます。

insert into foo (id) overriding system value 
values (1);

オプションgenerated by defaultを使うとき、これは本質的に既存のserialの実装と同じ振る舞いです:

create table foo 
(
  id integer generated by default as identity
);

値が手動で提供されるとき、基礎となるシーケンスも手動で調整される必要があります - serialカラムの場合と同じです。


識別列は、デフォルトでは主キーではありません(serial列と同じ)。 1にする必要がある場合は、主キー制約を手動で定義する必要があります。

33

すみません、古い質問を書き直すために、これはグーグルでポップアップした最初のStack Overflowの質問/答えでした。

この記事(Googleで最初に登場しました)は、PostgreSQL 10のより新しい構文の使用について話しています。 https://blog.2ndquadrant.com/postgresql-10-identity-columns/

それは偶然です:

CREATE TABLE test_new (
    id int GENERATED BY DEFAULT AS IDENTITY PRIMARY KEY,
);

それが役立つことを願っています:)

20
A L

SERIALまたはsequenceフィールドに直接挿入しないように注意する必要があります。そうしないと、シーケンスが挿入された値に達すると書き込みが失敗します。

-- Table: "test"

-- DROP TABLE test;

CREATE TABLE test
(
  "ID" SERIAL,
  "Rank" integer NOT NULL,
  "GermanHeadword" "text" [] NOT NULL,
  "PartOfSpeech" "text" NOT NULL,
  "ExampleSentence" "text" NOT NULL,
  "EnglishGloss" "text"[] NOT NULL,
  CONSTRAINT "PKey" PRIMARY KEY ("ID", "Rank")
)
WITH (
  OIDS=FALSE
);
-- ALTER TABLE test OWNER TO postgres;
 INSERT INTO test("Rank", "GermanHeadword", "PartOfSpeech", "ExampleSentence", "EnglishGloss")
           VALUES (1, '{"der", "die", "das", "den", "dem", "des"}', 'art', 'Der Mann küsst die Frau und das Kind schaut zu', '{"the", "of the" }');


 INSERT INTO test("ID", "Rank", "GermanHeadword", "PartOfSpeech", "ExampleSentence", "EnglishGloss")
           VALUES (2, 1, '{"der", "die", "das"}', 'pron', 'Das ist mein Fahrrad', '{"that", "those"}');

 INSERT INTO test("Rank", "GermanHeadword", "PartOfSpeech", "ExampleSentence", "EnglishGloss")
           VALUES (1, '{"der", "die", "das"}', 'pron', 'Die Frau, die nebenen wohnt, heißt Renate', '{"that", "who"}');

SELECT * from test; 
16
trev

尋ねられた質問の文脈において、そして@ sereja1cによるコメントに答えて、SERIALを作成することは暗黙のうちにシーケンスを作成します。

CREATE TABLE foo (id SERIAL,bar varchar);

CREATE TABLEは、シリアル列foo_id_seqのシーケンスfoo.idを暗黙的に作成します。したがって、SERIAL [4 Bytes]は、idに特定のデータ型が必要でない限り、使いやすさの点で優れています。

15
Prince

この方法は確かにうまくいくでしょう。

CREATE TABLE fruits(
   id SERIAL PRIMARY KEY,
   name VARCHAR NOT NULL
);

INSERT INTO fruits(id,name) VALUES(DEFAULT,'Apple');

or

INSERT INTO fruits VALUES(DEFAULT,'Apple');

次のリンクで詳細を確認できます。 http://www.postgresqltutorial.com/postgresql-serial/

2
webtechnelson

PostgreSQL 10以降

CREATE TABLE test_new (
    id int GENERATED BY DEFAULT AS IDENTITY PRIMARY KEY,
    payload text
);