PostgreSQL自动增量

Ian*_*Ian 556 postgresql auto-increment

我正在从MySQL切换到PostgreSQL,并想知道如何做自动增量值.我在PostgreSQL文档中看到了一个数据类型"serial",但是在使用它时会出现语法错误(在v8.0中).

Tre*_*rey 679

是的,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
Run Code Online (Sandbox Code Playgroud)

SERIAL只是序列周围的创建表时间宏.您不能将SERIAL更改为现有列.

  • 引用表名是一种习惯,因为我继承了一个具有混合案例名称的DB,并且引用表名是一种使用要求. (71认同)
  • 因为除非你有一个表`"Table"`和``table``然后只是将它不加引号并将其规范化为`table`.该约定绝不会在Pg中使用引号.如果需要,您可以使用混合大小写的名称来表示外观,只是不要求它:`CREATE TABLE fooBar(..); SELECT*FROM fooBar;`将起作用,`SELECT*FROM foobar`也是如此. (26认同)
  • @Evan Carroll - 为什么这是一个坏习惯(只是问)? (25认同)
  • 每个postgres doc,要么始终引用,要么不引用:http://www.postgresql.org/docs/current/interactive/sql-syntax-lexical.html#SQL-SYNTAX-IDENTIFIERS (23认同)
  • 引用表名是一种非常糟糕的做法 (18认同)
  • @EvanCarroll请不要引用`INSERT INTO user`. (10认同)
  • 加上...大多数关系数据库实现的标准是没有引号.如果有必要,它可以更简单地交换数据库实现. (7认同)
  • 引用没有错.实际上,它是Postgres,这是一个奇怪的问题,其中有一个无框架的架构约定.从任何其他数据库迁移都需要这样做. (4认同)
  • 虽然这个答案多年来一直是正确的,但从 Postgres 10 开始,您可以使用“IDENTITY”代替,如 @a_horse_with_no_name 在 /sf/answers/3292467531/ 中所述 (3认同)
  • 正如他所说,它只是一个围绕序列的宏.SERIAL类型只是一个整数和一个序列,列默认为序列的下一个值.与MySQL不同的是,没有什么特别的东西和voodooey继续下去. (2认同)
  • 埃文卡罗尔:如果你有一张桌子“桌子”和一张桌子“桌子”,我会怀疑你有一些有问题的自虐倾向,我建议你去看精神科医生。 (2认同)

Ahm*_*mad 222

您可以使用任何其他整数数据类型,例如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;
Run Code Online (Sandbox Code Playgroud)

最好使用您自己的数据类型,而不是用户串行数据类型.

  • @JayC:来自[文档](http://www.postgresql.org/docs/9.1/static/datatype-numeric.html#DATATYPE-SERIAL):_最后,序列被标记为"拥有"列,所以如果删除列或表,它将被删除._ (12认同)
  • 我说这实际上是更好的答案,因为它允许我通过设置默认列来修改我刚刚在PostgreSQL中创建的表(在阅读`CREATE SEQUENCE`之后http://www.postgresql.org/docs/ 8.1/interactive/sql-createsequence.html).但是,我不太清楚为什么你改变了主人. (11认同)
  • postgres社区为什么不重新发明autoincrement关键字? (7认同)
  • 如果你只想要一个较小的数据类型,还有一些小序列. (4认同)
  • @Dr Deo:他们使用serial而不是autoincrement关键字,我不知道为什么:) (2认同)
  • 为什么使用自己的数据类型更好?在当前的实现中,使用 SERIAL 与使用整数类型执行的操作相同:https://www.postgresql.org/docs/9.1/datatype-numeric.html#DATATYPE-SERIAL (2认同)
  • 我一直不明白为什么有些开发人员会重新发明轮子或使事情复杂化。如果 Postgres 已经有一个专门针对这个问题优化和创建的内部机制(串行),为什么要通过创建序列来让一切变得更加复杂呢? (2认同)

小智 105

如果要在已存在的表中向id添加序列,可以使用:

CREATE SEQUENCE user_id_seq;
ALTER TABLE user ALTER user_id SET DEFAULT NEXTVAL('user_id_seq');
Run Code Online (Sandbox Code Playgroud)

  • @Green:AUTO_INCREMENT不是SQL标准的一部分,它特定于MySQL.序列是在PostgreSQL中执行类似工作的东西. (23认同)
  • 如果你使用'id SERIAL',它会自动在PostgreSQL中创建一个序列.该序列的名称将是<table name> _ <column name> _seq (5认同)

Pro*_*ter 40

虽然看起来序列等同于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;
Run Code Online (Sandbox Code Playgroud)

你应该得到以下输出:

 uid | col_b 
-----+-------
   1 |     1
   3 |     2
(2 rows)
Run Code Online (Sandbox Code Playgroud)

注意uid如何从1变为3而不是1到2.

如果您使用以下方法手动创建自己的序列,则仍会出现这种情况

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;
Run Code Online (Sandbox Code Playgroud)

如果您想测试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);
Run Code Online (Sandbox Code Playgroud)

你应该得到以下没有fragementation:

+-----+-------+
| uid | col_b |
+-----+-------+
|   1 |     1 |
|   2 |     2 |
+-----+-------+
2 rows in set (0.00 sec)
Run Code Online (Sandbox Code Playgroud)

2.手动设置串行列值可能导致将来的查询失败.

@trev在之前的回答中指出了这一点.

要模拟此操作,请手动将uid设置为4,以便稍后"冲突".

INSERT INTO table1 (uid, col_b) VALUES(5, 5);
Run Code Online (Sandbox Code Playgroud)

表数据:

 uid | col_b 
-----+-------
   1 |     1
   3 |     2
   5 |     5
(3 rows)
Run Code Online (Sandbox Code Playgroud)

运行另一个插入:

INSERT INTO table1 (col_b) VALUES(6);
Run Code Online (Sandbox Code Playgroud)

表数据:

 uid | col_b 
-----+-------
   1 |     1
   3 |     2
   5 |     5
   4 |     6
Run Code Online (Sandbox Code Playgroud)

现在,如果你运行另一个插入:

INSERT INTO table1 (col_b) VALUES(7);
Run Code Online (Sandbox Code Playgroud)

它将失败,并显示以下错误消息:

错误:重复键值违反唯一约束"table1_pkey"DETAIL:键(uid)=(5)已存在.

相比之下,MySQL将优雅地处理这个,如下所示:

INSERT INTO table1 (uid, col_b) VALUES(4, 4);
Run Code Online (Sandbox Code Playgroud)

现在插入另一行而不设置uid

INSERT INTO table1 (col_b) VALUES(3);
Run Code Online (Sandbox Code Playgroud)

查询没有失败,uid只跳到5:

+-----+-------+
| uid | col_b |
+-----+-------+
|   1 |     1 |
|   2 |     2 |
|   4 |     4 |
|   5 |     3 |
+-----+-------+
Run Code Online (Sandbox Code Playgroud)

测试是在MySQL 5.6.33上进行的,适用于Linux(x86_64)和PostgreSQL 9.4.9

  • 你进行了比较,但我没有看到任何解决方案!这是答案吗? (9认同)
  • @Anwar它只是扩展了各种答案,说明答案是使用序列/序列.这提供了一些需要考虑的重要背景. (4认同)
  • 对于拥有 10 年 MSSQL 和 Mysql 经验的人来说,第 2 点是完全荒谬的。用户的随机手动插入可能会完全破坏应用程序 (3认同)

a_h*_*ame 36

从Postgres 10开始,还支持SQL标准定义的标识列:

create table foo 
(
  id integer generated always as identity
);
Run Code Online (Sandbox Code Playgroud)

除非明确要求,否则会创建一个无法覆盖的标识列.以下插入将失败,并且列定义为generated always:

insert into foo (id) 
values (1);
Run Code Online (Sandbox Code Playgroud)

但是,这可以推翻:

insert into foo (id) overriding system value 
values (1);
Run Code Online (Sandbox Code Playgroud)

使用该选项时,generated by default这与现有serial实现基本相同:

create table foo 
(
  id integer generated by default as identity
);
Run Code Online (Sandbox Code Playgroud)

手动提供值时,也需要手动调整基础序列 - 与serial列相同.


默认情况下,标识列不是主键(就像serial列一样).如果它应该是1,则需要手动定义主键约束.

  • @alex https://wiki.postgresql.org/wiki/Don't_Do_This#Don.27t_use_serial (5认同)
  • 这是截至 2021 年的正确答案 (2认同)

A L*_*A L 23

很抱歉,重新提出一个旧问题,但这是Google上出现的第一个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,
);
Run Code Online (Sandbox Code Playgroud)

希望有帮助:)

  • @adriaan 实际上“GENERATED … AS IDENTITY”命令是标准 SQL。首先在 *SQL:2003* 中添加,然后在 *SQL:2008* 中进行澄清。请参阅功能 # T174 &amp; F386 &amp; T178。 (2认同)

小智 16

您必须注意不要直接插入SERIAL或序列字段,否则当序列达到插入值时,写入将失败:

-- 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; 
Run Code Online (Sandbox Code Playgroud)


Pri*_*nce 15

在问题的上下文中以及@ sereja1c对评论的回复中,创建SERIAL隐式创建序列,因此对于上述示例 -

CREATE TABLE foo (id SERIAL,bar varchar);
Run Code Online (Sandbox Code Playgroud)

CREATE TABLE会隐式创建foo_id_seq串行列的序列foo.id.因此,SERIAL除非您需要特定的id数据类型,否则[4 Bytes]的易用性很好.


小智 8

从 PostgreSQL 10 开始

CREATE TABLE test_new (
    id int GENERATED BY DEFAULT AS IDENTITY PRIMARY KEY,
    payload text
);
Run Code Online (Sandbox Code Playgroud)


web*_*son 5

这种方式肯定会起作用,我希望它有所帮助:

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');
Run Code Online (Sandbox Code Playgroud)

您可以在下一个链接中查看详细信息:http : //www.postgresqltutorial.com/postgresql-serial/