以下两段代码是否完全相同,或者本地mysqli
事务函数是否会执行一些额外的操作?
我特别感兴趣的是在启动和提交事务/创建和释放保存点时如何处理和报告错误(如果有的话)。
使用特定的mysqli
函数进行交易处理:
$db = new mysqli('localhost', 'root', 'batman', 'batcave');
if (!$db->begin_transaction()) { throw new Exception($db->error); }
exc_query('INSERT INTO utility_belt SET item="apple"');
exc_query('INSERT INTO utility_belt SET item="banana"');
if (!$db->savepoint('vegetables')) { throw new Exception($db->error); }
exc_query('INSERT INTO utility_belt SET item="potato"');
exc_query('DELETE FROM utility_belt WHERE item="turnip"');
if (!$db->release_savepoint('vegetables')) { throw new Exception($db->error); }
exc_query('INSERT INTO utility_belt SET item="orange"');
if (!$db->commit()) { throw new Exception($db->error); }
使用普通的旧query
函数执行相同的操作:
$db = new mysqli('localhost', 'root', 'batman', 'batcave');
exc_query('START TRANSACTION');
exc_query('INSERT INTO utility_belt SET item="apple"');
exc_query('INSERT INTO utility_belt SET item="banana"');
exc_query('SAVEPOINT vegetables');
exc_query('INSERT INTO utility_belt SET item="potato"');
exc_query('DELETE FROM utility_belt WHERE item="turnip"');
exc_query('RELEASE SAVEPOINT vegetables');
exc_query('INSERT INTO utility_belt SET item="orange"');
exc_query('COMMIT');
上面使用的exc_query
函数的实现:
function exc_query($q) {
global $db;
if (!$db->query($q)) {
throw new Exception($db->error);
}
}
我为什么要问?
我在尝试RELEASE SAVEPOINT <name>
并看到SAVEPOINT <name> does not exist
时偶尔会遇到问题,即使我确定我之前已将SAVEPOINT <name>
发送到服务器。
根据我在MariaDB's documentation中读到的内容,如果我的示例中的初始START TRANSACTION
失败,则可能会抛出此错误。在这种情况下,不会启动事务,因此会{x}忽略SAVEPOINT <name>
,但RELEASE SAVEPOINT <name>
会随后抛出错误。
让我想知道的是,在这些情况下,以下行没有捕获创建创建初始事务的任何失败:
if (!$db->query($q)) {
throw new Exception($db->error);
}
......所以我不确定我是不对。不幸的是,关于mysqli函数如何工作的官方PHP文档是not very helpful。
答案 0 :(得分:0)
基于https://github.com/php/php-src/blob/master/ext/mysqli/mysqli_nonapi.c#L1115:
static int mysqli_savepoint_libmysql(MYSQL * conn, const char * const name, zend_bool release)
{
int ret;
char * query;
unsigned int query_len = spprintf(&query, 0, release? "RELEASE SAVEPOINT `%s`":"SAVEPOINT `%s`", name);
ret = mysql_real_query(conn, query, query_len);
efree(query);
return ret;
}
PHP只是向数据库发送完全相同的查询,并没有其他任何内容。
至于为什么MySQLi::savepoint()
与MySQLi::query("SAVEPOINT")
有任何不同,我不知道。 PHP源代码中的错误处理似乎是相同的,所以如果出现问题,我希望$db->query($q)
的结果为false
。