我正在从MySQL切换到PostgreSQL,想知道如何实现自动增量值。我在PostgreSQL文档中看到了一个数据类型“ serial”,但是在使用它(在v8.0中)时出现语法错误。
我正在从MySQL切换到PostgreSQL,想知道如何实现自动增量值。我在PostgreSQL文档中看到了一个数据类型“ serial”,但是在使用它(在v8.0中)时出现语法错误。
Answers:
是的,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更改为现有列。
"Table"
并"table"
把它留它不带引号的,它向规范化table
。惯例是绝对不要在Pg中使用引号。您可以根据需要使用大小写混合的名称来表示外观,只是不需要:CREATE TABLE fooBar ( .. ); SELECT * FROM fooBar;
可以,也可以SELECT * FROM foobar
。
您可以使用其他任何整数数据类型,例如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;
最好使用您自己的数据类型,而不是用户串行数据类型。
CREATE SEQUENCE
postgresql.org/docs/8.1/interactive/sql-createsequence.html上阅读后) 。但是,我不太确定您为什么更改所有者。
如果要将序列添加到已存在的表中的id,可以使用:
CREATE SEQUENCE user_id_seq;
ALTER TABLE user ALTER user_id SET DEFAULT NEXTVAL('user_id_seq');
ALTER COLUMN user_id
吗?
ERROR: syntax error at or near "DEFAULT"
任何建议吗?
看起来序列与MySQL auto_increment 等效,但是有一些细微但重要的区别:
失败查询后,串行列将增加。这导致失败查询的碎片化,而不仅仅是行删除。例如,在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)
@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上执行的
从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
列一样)。如果应为一个,则需要手动定义主键约束。
抱歉,请重新提出一个旧问题,但这是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,
);
希望有帮助:)
GENERATED … AS IDENTITY
命令是标准SQL。首先在SQL:2003中添加,然后在SQL:2008中进行澄清。参见功能#T174&F386&T178。
您必须注意不要直接插入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;
这种方法肯定会起作用,我希望它会有所帮助:
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 10开始
CREATE TABLE test_new (
id int GENERATED BY DEFAULT AS IDENTITY PRIMARY KEY,
payload text
);