返回介绍

I. 教程

II. SQL 语言

III. 服务器管理

IV. 客户端接口

V. 服务器端编程

VI. 参考手册

VII. 内部

VIII. 附录

SPI_execute

发布于 2019-09-30 03:09:49 字数 4543 浏览 877 评论 0 收藏 0

函数名

SPI_execute--执行一条命令

原型

int SPI_execute(const char * command, bool read_only, long count)

描述

SPI_execute 执行声明的 SQL 命令获取 count 行。如果 read_onlytrue ,命令必须是只读的,因此可以略微降低一些执行的开销。

这个函数只能在已连接的过程中调用。

如果 count 是零,则在命令适合的所有行上执行。如果 count 大于 0 ,那么命令执行的行数将被限制(很像一个 LIMIT 子句)。比如,

SPI_execute("INSERT INTO foo SELECT * FROM bar", false, 5);

将只允许最多 5 行插入表中。

你可以在一个字符串里传递多个命令。SPI_execute 返回最后执行的命令的结果。count 的限制独立地应用于每一个命令,但是不会应用于规则生成的隐藏命令。

如果 read_onlyfalseSPI_execute 递增命令计数器并且在执行字符串里的每个命令之前计算一个新的快照。如果当前事务的隔离级别是 SERIALIZABLE ,这个快照实际上并不改变,但是在 READ COMMITTED 模式里,这个快照更新允许每个命令看到其它会话的事务提交的结果。这样实际上是为了修改数据库的命令有一致的行为。

如果 read_onlytrueSPI_execute 并不更新快照或者命令计数器,并且它指允许简单的 SELECT 命令出现在命令字符串里。这个命令使用为周围的查询建立起来的快照执行。这个执行模式比读/写模式执行得略微块些,因为它消除了每个命令的一些开销。并且它还允许制作真正的稳定函数:因为随后的执行都将使用同一个快照,结果里不会有改变。

通常,在同一个使用 SPI 的函数里混杂只读和读写命令是不明智的;那样可能导致非常混乱的行为,因为只读的查询不能看到任何读写的查询做的数据库更新。

(最后)一条命令执行返回的结果的实际行数会放在全局的变量 SPI_processed 里。如果函数的返回值是 SPI_OK_SELECT, SPI_OK_INSERT_RETURNING, SPI_OK_DELETE_RETURNING, SPI_OK_UPDATE_RETURNING ,那么你可以使用全局指针 SPITupleTable *SPI_tuptable 访问结果行。一些实用命令(比如 EXPLAIN)还返回行集合,并且 SPI_tuptable 也将在这种情况下包含结果。

结构 SPITupleTable 是这样定义的:

typedef struct
{
    MemoryContext tuptabcxt;    /* 结果表的内存环境 */
    uint32      alloced;        /* 分配的 vals 数目 */
    uint32      free;           /* 空闲的 vals 数目 */
    TupleDesc   tupdesc;        /* 行描述符 */
    HeapTuple  *vals;           /* 行 */
} SPITupleTable;

vals 是一个指向数据行的的指针数组(有效记录的数目由 SPI_processed 给出)。tupdesc 是一个行描述符,你可以传递给 SPI 函数处理这些数据行。tuptabcxt, alloced, free 是 SPI 的内部字段,并非给 SPI 调用者使用的。

SPI_finish 释放所有在当前过程中分配的 SPITupleTable 。如果你已经处理完特定的结果表,那么可以更早地释放它,方法是调用 SPI_freetuptable

参数

const char * command

包含要执行的命令的字符串

bool read_only

true 用于只读的执行

long count

处理或返回的最大行数

返回值

如果命令执行成功,那么返回下列值之一(非负数):

SPI_OK_SELECT

如果执行了一个 SELECT 但不是 SELECT INTO

SPI_OK_SELINTO

如果执行了一条 SELECT INTO

SPI_OK_INSERT

如果执行了一条 INSERT

SPI_OK_DELETE

如果执行了一条 DELETE

SPI_OK_UPDATE

如果执行了一条 UPDATE

SPI_OK_INSERT_RETURNING

如果执行了一条 INSERT RETURNING

SPI_OK_DELETE_RETURNING

如果执行了一条 DELETE RETURNING

SPI_OK_UPDATE_RETURNING

如果执行了一条 UPDATE RETURNING

SPI_OK_UTILITY

如果执行了一条实用命令(比如 CREATE TABLE)

发生错误时,返回下列负数值之一:

SPI_ERROR_ARGUMENT

如果 commandNULLcount 小于 0

SPI_ERROR_COPY

如果企图进行 COPY TO stdoutCOPY FROM stdin

SPI_ERROR_CURSOR

如果企图进行 DECLARE, CLOSE, FETCH

SPI_ERROR_TRANSACTION

如果企图调用任何事务操纵命令(BEGIN, COMMIT, ROLLBACK, SAVEPOINT, PREPARE TRANSACTION, COMMIT PREPARED, ROLLBACK PREPARED 或它们的变种)

SPI_ERROR_OPUNKNOWN

命令类型未知(不应该发生)

SPI_ERROR_UNCONNECTED

如果从一个未连接的过程中调用

注意

SPI_execute, SPI_exec, SPI_execute_plan, SPI_execp 函数修改 SPI_processedSPI_tuptable (只是一个指针,不是结构的内容)。如果你需要跨越后面的调用访问 SPI_execute 或者相关函数的结果表,那么需要把这两个全局俩保存到一个局部过程变量中。

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

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

发布评论

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