PostgreSQL自动增量


578

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


9
如果您要提供查询和错误信息,则可能会有人告诉您查询出了什么问题。

2
我的第一个热门歌曲是“ Mich”,因为这个问题引起了足够多的关注,因此为什么不投票呢?PS:如果您不知道该怎么做,这并非微不足道。
baash05 2012年

1
如果您的客户端驱动程序是Npgsql,则SERIAL是首选。提供程序在使用SELECT currval(pg_get_serial_sequence('table','column'))进行INSERT之后在内部选择新值。如果基础列不是序列类型(例如,数字类型+显式序列),则此操作将失败
Olivier MATROT 2012年

只是出于好奇...为什么有人必须从非常好的MySQL迁移到PostgreSql?
villamejia '16

17
...甚至更好。
罗默

Answers:


701

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


19
引用表名是一个非常糟糕的做法
Evan Carroll'2

71
引用表名是一种习惯,因为我继承了混合大小写名称的数据库,并且引用表名是必须使用的。
Trey 2010年

26
@Evan Carroll-为什么这是个坏习惯(只是问问)?
基督教徒

27
因为除非你有一个表,"Table""table"把它留它不带引号的,它向规范化table。惯例是绝对不要在Pg中使用引号。您可以根据需要使用大小写混合的名称来表示外观,只是不需要:CREATE TABLE fooBar ( .. ); SELECT * FROM fooBar;可以,也可以SELECT * FROM foobar
埃文·卡罗尔

26
:每Postgres的文档,无论是持续报价或所享有postgresql.org/docs/current/interactive/...
Καrτhικ

226

您可以使用其他任何整数数据类型,例如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;

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


11
我说这实际上是更好的答案,因为它允许我通过将列设置为默认值来修改刚在PostgreSQL中创建的表(在CREATE SEQUENCE postgresql.org/docs/8.1/interactive/sql-createsequence.html上阅读后) 。但是,我不太确定您为什么更改所有者。
JayC 2011年

12
@JayC:从文档开始最后,该序列被标记为该列“拥有者”,因此如果删除该列或表,该序列将被删除。
user272735 2012年

8
为什么postgres社区不只是重新发明了autoincrement关键字?
Deo博士

2
@Dr Deo:他们使用串行而不是autoincrement关键字,我不知道为什么:)
Ahmad

4
如果您只想使用较小的数据类型,也可以使用smallserial。
贝尔达兹

110

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

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

什么是顺序?AUTO_INCREMENT在哪里?
格林

23
@绿色:AUTO_INCREMENT不是SQL标准的一部分,它特定于MySQL。序列在PostgreSQL中起到类似的作用。
贝尔达兹

5
如果使用“ id SERIAL”,它将在PostgreSQL中自动创建一个序列。该序列的名称为<表名> _ <列名> _seq
Jude Niroshan 2015年

你不用用ALTER COLUMN user_id吗?
亚历克

我尝试了这种方法,但出现错误:有ERROR: syntax error at or near "DEFAULT"任何建议吗?
伊利·菲亚科夫

44

看起来序列与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”详细信息:键(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 |
+-----+-------+

测试是针对Linux(x86_64)和PostgreSQL 9.4.9在MySQL 5.6.33上执行的


10
您在进行比较,但在这里看不到任何解决方案!是答案吗?
安瓦尔

4
@Anwar只是扩展了各种答案,指出答案是使用序列/序列。这提供了一些重要的考虑因素。
Programster

38

从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列一样)。如果应为一个,则需要手动定义主键约束。


26

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

希望有帮助:)


1
确实,这是在PostgreSQL 10中使用的方法,并且与其他数据库软件(如DB2或Oracle)的语法相同。
adriaan

1
@adriaan实际上,这些GENERATED … AS IDENTITY命令是标准SQL。首先在SQL:2003中添加,然后在SQL:2008中进行澄清。参见功能#T174&F386&T178。
罗勒·布尔克

16

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

15

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

CREATE TABLE foo (id SERIAL,bar varchar);

CREATE TABLEfoo_id_seq为序列列隐式创建序列foo.id。因此,SERIAL[4字节]易于使用,除非您的ID需要特定的数据类型。


3

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

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/


By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.