2

首先是有罪的代码:

$preparedGetData = $db->prepare("CALL getData(?)");
foreach($userSet as $r) {
    $preparedGetData->execute(array($r['id_user']));
    $rs = $preparedGetData->fetchAll();
    $preparedGetData->closeCursor();
}

解释

getData= mysql 中的存储过程
$db= Zend_Db_Adapter_Pdo_Mysql 的实例(使用 Zend 版本 1.10.0 )

症状

  • closeCursor当我在第二个周期中 退出时,我已经收到错误:

    PHP 致命错误:未捕获异常 'PDOException' 并带有消息 'SQLSTATE[HY000]:一般错误:2014 在其他无缓冲查询处于活动状态时无法执行查询。考虑使用 PDOStatement::fetchAll()。或者,如果您的代码只针对 mysql 运行,您可以通过设置 PDO::MYSQL_ATTR_USE_BUFFERED_QUERY 属性来启用查询缓冲。

  • 但我正在使用fetchAll

  • 当我添加closeCursor结果时,结果不完整。在查询浏览器中调用CALL getData('3872')返回 1 行 72 列。上面的代码所做的相同返回只有前 41 列的 1 行。

我做错了什么?

Edit2:半解决方案

代码更新为:

$preparedGetData = $db->prepare("CALL getData(?)");
foreach($userSet as $r) {
    $preparedGetData->execute(array($r['id_user']));

    $rs=array();

    do {
        try {
            $partial_rowset = $preparedGetData->fetchAll(); // When I put fetchAll() after the end of the cycle, I get empty resultset.
        } catch (PDOException $error) { // The PDOException doesn't get caught here. Why?
            error_log($error);
        } catch (Exception $error) { 
            error_log($error);
        }
        if ($partial_rowset) {
            $rs=array_merge($rs,$partial_rowset);
        }
    } while ($preparedGetData->nextRowset());
}

症状

  • 出现错误:
    PHP 致命错误:未捕获的异常 'PDOException' 带有消息 'SQLSTATE[HY000]:一般错误
    下一个异常 'Zend_Db_Statement_Exception' 带有消息 'SQLSTATE[HY000]:/home/GIT/includes/Zend/Db/ 中的一般错误' Statement/Pdo.php:294
    fetchAll.
  • 我可以使用通用异常捕获此错误。
  • 使用此代码,我得到所有 72 列。
  • 我认为这很讨厌,因为我故意捕获通用异常并将其转换为日志。我猜这也将成为一个性能问题(循环运行大约 10 000 次)。
4

1 回答 1

2

MySQL 存储过程可能有多个结果集。也就是说,您可以在一个存储过程中运行多个 SELECT 查询,然后遍历这些多个结果集。

这使获取变得复杂,因为整个结果实际上就像一个数组数组(多个结果集,每个结果集可能有多行,每行可能有多个列)。

但只从一个结果集中fetchAll()获取所有行。所以当你调用一个存储过程时,你需要强制它刷新所有的结果集。也就是说,继续调用,直到该函数返回 null。nextRowset()

Zend_Db API 是根据 PDO 建模的,因此您可以nextRowset()PDOStatement::nextRowset().

不幸的是,没有像fetchAllRowsets(). 你可以自己做一个while循环。

于 2013-10-23T23:20:19.370 回答