ThinkChat2.0新版上线,更智能更精彩,支持会话、画图、阅读、搜索等,送10W Token,即刻开启你的AI之旅 广告
# DECLARE ## Name DECLARE -- 定义一个游标 ## Synopsis ``` DECLARE _name_ [ BINARY ] [ INSENSITIVE ] [ [ NO ] SCROLL ] CURSOR [ { WITH | WITHOUT } HOLD ] FOR _query_ ``` ## 描述 `DECLARE`允许用户创建游标,用于在一个大的查询里面检索少数几行数据。 在游标建立后,使用[FETCH](#calibre_link-74)可以从中取出行。 > **Note:** 本页描述游标在SQL命令级别的使用。如果你试图在一个PL/pgSQL函数里面使用游标, 结果往往是不同的;参阅[Section 40.7](#calibre_link-491)。 ## 参数 `_name_` 将要创建的游标名。 `BINARY` 令游标以二进制而不是文本格式返回数据。 `INSENSITIVE` 表明从游标检索出来的数据不应该被发生在游标创建后的游标的更新动作影响。 在PostgreSQL里,这是缺省行为;这个关键字没有什么作用, 只在为了和 SQL 标准兼容的时候接受它。 `SCROLL``NO SCROLL` `SCROLL`声明该游标可以用于以倒序的方式检索数据行(也就是反向检索)。 根据查询的执行计划的不同,声明`SCROLL`可能会对查询的执行时间有不良影响。 `NO SCROLL`声明该游标不能用于以倒序的方式检索数据行。 缺省仅允许在某些情况下倒序检索,这不同于指定`SCROLL`。 参见[_注意_](#calibre_link-492)获取细节。 `WITH HOLD` `WITHOUT HOLD` `WITH HOLD`(缺省)声明该游标可以在创建它的事务成功提交后继续使用。 `WITHOUT HOLD`声明该游标不能在创建它的事务之外使用。 `_query_` 一个[SELECT](#calibre_link-104)或[VALUES](#calibre_link-106)命令,它提供游标返回的行。 `BINARY`, `INSENSITIVE`, `SCROLL` 关键字可以以任何顺序出现。 ## 注意 通常游标和`SELECT`一样返回文本格式。`BINARY` 选项声明游标应该返回二进制格式的数据。这样就减少了服务器和客户端的转化工作, 花费程序员更多的工作解决平台依赖的二进制数据格式问题。 比如,如果查询从某个整数列返回 1 ,在缺省的游标里将获得一个字符串`1`, 但在二进制游标里将得到一个 4 字节的包含该数值内部形式的数值(大端顺序)。 应该小心使用二进制游标。一些客户端应用(比如psql) 是不能识别二进制游标的,它们期望返回的数据是文本格式。 > **Note:** 如果客户端应用使用"扩展查询"协议发出`FETCH`命令, 那么 Bind 协议声明数据是用文本还是用二进制格式检索。这个选择覆盖游标的定义。 因此,在使用扩展查询协议的时候,二进制游标的概念已经过时了, 任何游标都可以当作文本或者二进制的格式发出。 如果没有声明`WITH HOLD`,那么这个命令创建的游标只能在当前事务中使用。 这样,不带`WITH HOLD`的`DECLARE`在事务块外面没有任何用处: 游标将一直存活到事务结束。因此,PostgreSQL 将在这样的命令出现在事务块外面的时候报错。使用[BEGIN](#calibre_link-493), [COMMIT](#calibre_link-494)和[ROLLBACK](#calibre_link-495)定义一个事务块。 如果声明了`WITH HOLD`并且创建该游标的事务成功提交, 那么游标还可以在同一会话随后的事务里访问。但如果创建它的事务回滚, 那么游标被删除。带`WITH HOLD`创建的游标是用一个明确的 `CLOSE`命令或者是会话终止来关闭的。在目前的实现里, 由一个游标代表的行是被拷贝到一个临时文件或者内存区里的, 这样他们就仍然可以在随后的事务中被访问。 当查询包含`FOR UPDATE`或`FOR SHARE`的时候, 不能指定`WITH HOLD`。 在定义一个要用来反向抓取的游标的时候,应该声明`SCROLL`选项, 这是 SQL 标准要求的。不过,为了和早期的版本兼容, 只要游标的查询计划简单得不需要额外的开销,PostgreSQL 在没有声明`SCROLL`的时候也允许反向抓取。不过, 建议应用开发人员不要依赖于使用没有使用`SCROLL` 定义的游标的反向查找功能。如果声明了`NO SCROLL` , 那么不管怎样都会禁止反向抓取的功能。 当查询包含`FOR UPDATE`或`FOR SHARE`时也不允许反向抓取; 因此在这种情况下不能声明`SCROLL`。 | **Caution** | |:--- | | 如果调用任何不稳定的函数,那么可以回滚并且`WITH HOLD` 游标可能给出意外的结果(参阅xref linkend="xfunc-volatility">)。 当重新抓取到以前抓取到的行时,函数可能会重新执行,可能导致结果与之前的不同。 一个绕开这种情况的方法是声明`WITH HOLD`游标, 并且在从它读取任何行之前提交事务。这将强制游标的整个输出在临时存储中物化, 这样不稳定的函数精确的每行只执行一次。 | 如果游标的查询包括`FOR UPDATE`或`FOR SHARE`, 那么返回的行在它们第一次被抓取的时候锁定,和有这个选项的[SELECT](#calibre_link-104) 一样。另外,返回行将是最新的版本;因此提供这些选项相当于SQL标准调用一个 "敏感的游标"。(和`FOR UPDATE`或 `FOR SHARE`一起指定`INSENSITIVE`是错误的。) | **Caution** | |:--- | | 如果游标定义为使用`UPDATE ... WHERE CURRENT OF`或 `DELETE ... WHERE CURRENT OF`,通常推荐使用`FOR UPDATE`。 使用`FOR UPDATE`阻止其他会话在抓取和更新之间改变行。没有`FOR UPDATE`, 如果在创建游标后改变了行,那么随后的`WHERE CURRENT OF`命令将没有作用。 使用`FOR UPDATE`的另外一个原因是,没有它,如果游标查询不符合SQL标准的 "简单可更新"原则,那么随后的`WHERE CURRENT OF`可能会失败 (尤其是,游标必须只引用一个表,并且不使用分组或`ORDER BY`)。 不是简单可更新的游标可能会也可能不会工作,取决于计划选择细节;所以在最坏的情况下, 应用可能在测试中工作而在生产中失败。 不和`WHERE CURRENT OF`一起使用`FOR UPDATE`的主要原因是, 你需要游标是可回滚的,或对随后的更新不敏感的(也就是说,持续显示旧的数据)。 如果需要这样,请注意上面显示的注意事项。 | SQL 标准中的游标只能在嵌入SQL(ESQL)的应用中使用。 PostgreSQL服务器没有一个明确的`OPEN` 语句;一个游标被认为在定义时就已经打开了。不过,PostgreSQL 嵌入的 SQL 预处理器(ECPG)支持 SQL 标准的习惯, 包括那些和`DECLARE`和`OPEN`相关的语句。 可以通过查询[`pg_cursors`](#calibre_link-73) 系统视图看到所有可用游标。 ## 例子 定义一个游标: ``` DECLARE liahona CURSOR FOR SELECT * FROM films; ``` 参阅[FETCH](#calibre_link-74)获取有关游标使用的更多例子。 ## 兼容性 SQL标准说它是依赖于实现的,不管游标对缺省的底层数据的当前更新是否敏感。 在PostgreSQL中,游标缺省是敏感的,并且可以通过声明 `FOR UPDATE`使其敏感。其他产品工作可能不同。 SQL 标准只允许在嵌入的SQL中和模块中使用游标。 PostgreSQL允许交互地使用游标。 二进制游标是PostgreSQL扩展。 ## 又见 [CLOSE](#calibre_link-69), [FETCH](#calibre_link-74), [MOVE](#calibre_link-75)