返回介绍

I. 教程

II. SQL 语言

III. 服务器管理

IV. 客户端接口

V. 服务器端编程

VI. 参考手册

VII. 内部

VIII. 附录

9.12. 序列操作函数

发布于 2019-09-30 03:06:23 字数 3851 浏览 1024 评论 0 收藏 0

本节描述 PostgreSQL 用于操作序列对象的函数。序列对象(也叫序列生成器或者就叫序列)都是用 CREATE SEQUENCE 创建的特殊的单行表。一个序列对象通常用于为行或者表生成唯一的标识符。在表9-34中列出的序列函数为我们从序列对象中获取后续的序列值提供了简单的、多用户安全的方法。

表9-34. 序列函数

函数返回类型描述
currval(regclass)bigint返回最近一次用 nextval 获取的指定序列的数值
nextval(regclass)bigint递增序列并返回新值
setval(regclass, bigint)bigint设置序列的当前数值
setval(regclass, bigint, boolean)bigint设置序列的当前数值以及 is_called 标志

被序列函数操作的序列是用 regclass 参数声明的,它只是序列在 pg_class 系统表里面的 OID 。不过,你不需要手工查找 OID ,因为 regclass 数据类型的输入转换器会帮你做这件事。只要写出单引号包围的序列名字即可,因此它看上去像文本常量。要达到和处理普通 SQL 名字的兼容性,这个字符串将转换成小写,除非在序列名字周围包含双引号,因此

nextval('foo')      操作序列号 foo
nextval('FOO')      操作序列号 foo
nextval('"Foo"')    操作序列号 Foo

必要时序列名可以用模式修饰:

nextval('myschema.foo')     操作 myschema.foo
nextval('"myschema".foo')   同上
nextval('foo')              在搜索路径中查找 foo

参阅节8.12获取有关 regclass 的更多信息。

【注意】在 PostgreSQL 8.1之前,序列函数的参数类型是 text 而不是 regclass ,而上面描述的从文本字符串到 OID 值的转换将在每次调用的时候发生。为了向下兼容,这个机制仍然存在,但是在内部实际上是在函数调用前隐含地将 text 转换成 regclass 实现的。

如果你把一个序列函数的参数写成一个无修饰的文本字符串,那么它将变成类型为 regclass 的常量。因为这只是一个 OID ,它将跟踪最初标识的序列,而不管后面是否改名、模式是否变化等等。这种"提前绑定"的行为通常是字段缺省和视图里面引用序列所需要的。但是有时候你可能想要"推迟绑定",这个时候序列的引用是在运行时解析的。要获取推迟绑定的行为,我们可以强制存储为 text 常量,而不是 regclass 常量:

nextval('foo'::text)      foo 在运行时查找

请注意,推迟绑定是 PostgreSQL 版本 8.1 之前唯一可用的行为,因此你可能需要在旧的应用里如此使用来保留旧有的语意。

当然,序列函数的参数也可以是表达式。如果它是一个文本表达式,那么隐含的转换将导致运行时的查找。

用的序列函数有:

nextval

递增序列对象到它的下一个数值并且返回该值。这个动作是自动完成的:即使多个会话并发运行 nextval ,每个进程也会安全地收到一个唯一的序列值。

currval

在当前会话中返回最近一次 nextval 抓到的该序列的数值。如果在本会话中从未在该序列上调用过 nextval ,那么会报告一个错误。请注意,因为此函数返回一个会话范围的数值,它也能给出一个可预计的结果,可以判断其它会话是否执行过 nextval 函数。

lastval

返回当前会话里最近一次 nextval 返回的数值。这个函数等效于 currval ,只是它不用序列名作为参数,它抓取当前会话里面最近一次 nextval 使用的序列。如果当前会话还没有调用过 lastval ,那么调用 nextval 是会报错的。

setval

重置序列对象的计数器数值。双参数的形式设置序列的 last_value 字段为声明数值并且将其 is_called 字段设置为 true ,表示下一次 nextval 将在返回数值之前递增该序列。在三参数形式里,is_called 可以设置为 truefalse 。如果你把它设置为 false ,那么下一次 nextval 将返回这里声明的数值,而从随后的 nextval 才开始递增该序列。比如:

SELECT setval('foo', 42);           下次 nextval 将返回 43
SELECT setval('foo', 42, true);     和上面一样
SELECT setval('foo', 42, false);    下次 nextval 将返回 42

setval 返回的结果就是它的第二个参数的数值。

如果一个序列对象是带着缺省参数创建的,那么对它调用 nextval 将返回从 1 开始的后续的数值。其它的行为可以通过使用 CREATE SEQUENCE 命令里的特殊参数获取;参阅其命令参考页获取更多信息。

【重要】为了避免从同一个序列获取数值的当前事务被阻塞,nextval 操作决不会回滚;也就是说,一旦一个数值已经被抓走,那么就认为它已经用过了,即使调用 nextval 的事务后面又退出了也一样。这就意味着退出的事务可能在序列赋予的数值中留下"空洞"。setval 操作也决不回滚。

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。
列表为空,暂无数据
    我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
    原文