PostgreSQL自动增量

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了PostgreSQL自动增量相关的知识,希望对你有一定的参考价值。

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

答案

是的,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更改为现有列。

另一答案

自PostgreSQL 10以来

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

你可以使用任何其他integer data type,例如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;

最好使用您自己的数据类型,而不是用户serial data type

另一答案

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

CREATE SEQUENCE user_id_seq;
ALTER TABLE user ALTER user_id SET DEFAULT NEXTVAL('user_id_seq');
另一答案

虽然看起来序列等同于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变为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;

如果您想测试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”DETAIL:键(uid)=(5)已存在。

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

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

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

INSERT INTO table1 (col_b) VALUES(3);

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

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

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

另一答案

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

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

除非明确要求,否则会创建一个无法覆盖的标识列。以下插入将失败,并且列定义为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,则需要手动定义主键约束。

另一答案

很抱歉,重新提出一个旧问题,但这是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,
);

希望有帮助:)

另一答案

您必须注意不要直接插入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; 
另一答案

在问题的背景下,@ sereja1c回复评论,创建SERIAL隐含地创建序列,所以对于上面的例子 -

CREATE TABLE foo (id SERIAL,bar varchar);

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

另一答案

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

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/

以上是关于PostgreSQL自动增量的主要内容,如果未能解决你的问题,请参考以下文章

从 postgresql 字段中删除自动增量

PostgreSQL中的自动增量列?

在 postgresql 中重置自动增量计数器

postgresql中的自定义自动增量字段(发票/订单号)

使用 idx 自动增量将数据帧插入到 postgresql sqlalchemy

Flyway - 自动增量 ID 不适用于 PostgreSQL 中的测试数据