如果不习惯简书的格式,可以点击下面链接查看:笔记版本
原文链接:@L_403_1@
什么是Sequence?在twitter(也可能是其他地方)上很多人在问关于如何在Postgresql中使用sequence。为了避免重复回答这个问题,我觉得在这里总结一下在Postgresql中使用sequence的基本步骤还是很有意义的。
Sequence是数据库中一类特殊的对象,其用于生成唯一数字标识符。一个典型的应用场景就是手动生成一系列主键。Sequence和MysqL中的auto_INCREMENT
的概念很像,但又不完全相同。
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
这个关键字。
下面是例子,二者完全等价:
-- 忽略IDINSERT 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()
返回的值,只有该用户的插入 *** 作才会更新该值。
使用上文中所说的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 *** 作是基于session的,与事务无关。nextval()
会递增Sequence的值,但是即便是事务回滚了Sequence也不会撤销,而不论是在事务内外,currval()
都会返回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()
这个函数。
见Postgresql文档:
Serial Types Sequence相关函数 新建Sequence 删除Sequence总结说明:本文是基于8.2或者更早的版本进行讲解的,翻译本文的时候Postgresql已经更新到了9.6,细节可能会有所不同。
以上是内存溢出为你收集整理的在Postgres里用Sequence[译]全部内容,希望文章能够帮你解决在Postgres里用Sequence[译]所遇到的程序开发问题。
如果觉得内存溢出网站内容还不错,欢迎将内存溢出网站推荐给程序员好友。
欢迎分享,转载请注明来源:内存溢出
评论列表(0条)