在Postgres里用Sequence[译]


原文链接:http://www.neilconway.org/docs/sequences/
如果不习惯简书的格式,可以点击下面链接查看:笔记版本

在twitter(也可能是其他地方)上很多人在问关于如何在PostgreSql中使用sequence。为了避免重复回答这个问题,我觉得在这里总结一下在Postgresql中使用sequence的基本步骤还是很有意义的。

什么是Sequence?

Sequence是数据库中一类特殊的对象,其用于生成唯一数字标识符。一个典型的应用场景就是手动生成一系列主键。Sequence和Mysql中的AUTO_INCREMENT的概念很像,但又不完全相同。

在表格中怎么使用Sequence?

Sequence最常见的场景就是生成Serial这个伪类型,Serial类型主要有下面几个特征:

  • Serial的值就是Sequence生成的。
  • 每次调用Sequence会生成一个新值(Serial类型的)。
  • 由于Sequence生成的值都是非空的,所以它会在这一列数据上加一个NOT NULL的标志。
  • Sequence是自动生成的,Postgresql假设Sequence只用于生成Series列(唯一),所以如果删除了这一列,数据库会自动删除这个Sequence。

例如:下面的命令会新建一个表和一个Sequence对象,并且把这个Sequence对象关联到这个表上。

test=# CREATE TABLE users (
test(#     id    SERIAL, -- assign each user a numeric ID
test(#     name  TEXT,
test(#     age   INT4
test(# );
NOTICE:  CREATE TABLE will create implicit sequence
"users_id_seq" for serial column "users.id"
CREATE TABLE

在这个例子里,自动生成的Sequence对象被自动命名为user_id_seq。如果不想让数据库自动命名的话,可以调用下面介绍的pg_get_serial_sequence()这个函数。

需要注意的是使用Series列并不会自动创建索引,也不会把这一列标注为主键。解决这个问题也很简单,只需要手动加上PRIMARY KEY这个标示即可。

CREATE TABLE users (
    -- make the "id" column a primary key; this also creates
    -- a UNIQUE constraint and a b+-tree index on the column
    id    SERIAL PRIMARY KEY,
    name  TEXT,
    age   INT4
);

怎么把Sequence的值赋给新插入的数据?

如果你用了Serial变量,其默认值就是Sequence下一次生成的值。为了让插入的时候取到这个默认值,要么忽略插入对象中Serial这一列的值,要么在这个位置写上DEFAULT这个关键字。
下面是例子,二者完全等价:

-- 忽略id
INSERT INTO users (name, age) VALUES ('Mozart', 20);
-- 使用DEFAULT关键字
INSERT INTO users (name, age, id) VALUES ('Mozart', 20, DEFAULT);

怎么获取Sequence最新生成的数据?

你可以使用currval()这个函数来获取Sequence最新生成的值,需要注意的是这里获取的是本次session中的值,这么设计是有目的的。currval()需要一个参数:Sequence的名字。可以调用pg_get_serial_sequence()来找到和指定列关联的Sequence的名字。
下面是例子:

SELECT currval(pg_get_serial_sequence('users', 'id'));

需要注意的是,如果该Sequence在本次session中从来没有生成过新的数据,则currval()会报错。

会不会有竞争存在?

假如一个数据库客户端插入了一个Sequence生成的值,与此同时另一个客户端又插入了一个值,这样currval()获取的值岂不是有可能取到一个错误的值?

事实上并不会这样,Sequence通过优雅的设计避免了这个问题。currval()这回返回当前session的新值,所以其他用户的插入并不会改变currval()返回的值,只有该用户的插入操作才会更新该值。

插入并获取插入的数据ID岂不是需要两个Query语句?

使用上文中所说的currval()我们需要执行两次Query语句:一个用于插入数据,另一个用于获取新插入的ID。为了减少和数据库连接的次数,我们可以把插入和获取ID的两条指令连起来一起丢到数据库执行(译者注:不熟悉PHP,不过个人不推荐这么做,毕竟不是原子操作,高并发长连接的情况下出错的可能性应该还是有的)。下面是例子(PHP,在nodejs和python里可以有类似的写法):

pg_exec("INSERT INTO users (name, age) VALUES ('Bach', 15);
         SELECT currval(pg_get_serial_sequence('users', 'id'));")

这条指令虽有两条命令,但是只需要和数据库通信一次即可,所以基本可以忽略第二条query的额外耗时。

另外,对于Postgresql 8.2以后的版本,可以直接采用RETURNING语法解决这个问题。(推荐这种方式)

INSERT INTO users (name, age) VALUES ('Liszt', 10) RETURNING id;

该语句会返回新插入的行的ID。

Sequence生成的数据的范围?

Sequence生成的是64位的整形数据,我们在上面用的Serial类型是一个32位的整形,如果需要使用64位的Serial则需要使用Serial8类型。

Sequence生成的数据会不连续吗?

当然会,Sequence是用于生成唯一的数据标识,并不需要严格连续。比如:如果两个客户端同时插入不同的数据(会调用nextval())的时候,每个客户端会得到一个新的Sequence值。如果其中一个在事务中失败或者其他原因回滚了,这个时候就会出现Sequence数据(即ID)不连续的现象,这只是其中一个例子。
修复这个问题也很简单,具体参考下面的链接解决ID不连续的问题

事务中的Sequence

Sequence操作是基于session的,与事务无关。nextval()会递增Sequence的值,但是即便是事务回滚了Sequence也不会撤销,而不论是在事务内外,currval()都会返回Sequence最新的值。

两个表共享一个Sequence?

实现这个功能的最简单做法就是手动生成一个Sequence,然后不要使用Serial类型,而是手动把Sequence绑定到对应的列。
下面是具体例子:

CREATE SEQUENCE common_fruit_id_seq;

CREATE TABLE apples (
    id      INT4 DEFAULT nextval('common_fruit_id_seq') NOT NULL,
    price   NUMERIC
);

CREATE TABLE oranges (
    id      INT4 DEFAULT nextval('common_fruit_id_seq') NOT NULL,
    weight  NUMERIC
);

nextval() 会生成一系列新的值。需要注意的是这个手动生成的Sequence并不会随着表被自动删除,而且你也没法调用 pg_get_serial_sequence()这个函数。

更多Sequence的资料

见Postgresql文档:


说明:本文是基于8.2或者更早的版本进行讲解的,翻译本文的时候Postgresql已经更新到了9.6,细节可能会有所不同。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,324评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,356评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,328评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,147评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,160评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,115评论 1 296
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,025评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,867评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,307评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,528评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,688评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,409评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,001评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,657评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,811评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,685评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,573评论 2 353

推荐阅读更多精彩内容