PHP PDO多选查询一致地删除最后一行

时间:2014-01-31 17:05:46

标签: php mysql pdo

我遇到了使用PDO语句进行多项选择的错误。

我正在构建一个包含许多SELECT的SQL查询,无论它做了多少个SELECT语句,都会删除最后一个行集。

以下是发生了什么的截断示例

$pdo = /* connection stuff here */
$sql = "select 1; select 2; select 3; select 4;";
$statement = $pdo->query($sql);

do {
    $rowset = $statement->fetchAll();
    if($rowset) {               
        // Do stuff with $rowset
    }       
} while($statement->nextRowset());

执行上述操作,1-3成功检索为行集,但不是4 我无法解释为什么会这样。使用相同的PDO对象进行任何后续查询会导致错误:

PDO::query(): SQLSTATE[HY000]: General error: 2014 Cannot execute queries while other unbuffered queries are active. 
Consider using PDOStatement::fetchAll(). Alternatively, if your code is only ever going to run against mysql, you may enable query buffering by setting the PDO::MYSQL_ATTR_USE_BUFFERED_QUERY attribute.

以上do ... while ...例程基于PHP nextRowset()函数http://us1.php.net/manual/en/pdostatement.nextrowset.php

上的文档中可以找到的内容

最后调用$statement->closeCursor()似乎无法正常工作

我使用的例程要复杂得多,但我可以确认sql的行为符合预期(通过使用PHPMyAdmin直接将其插入MySQL并使用mysqli->multi_query()运行它们,两者都返回预期结果)

我找到了一个有类似问题的人并发了一张PHP错误票,显然标记为已修复:https://bugs.php.net/bug.php?id=61207&edit=1

任何人都可以向我解释导致最后一行被删除的原因是什么?谢谢!

版本: PHP 5.4.12 MySQL 5.6.12

编辑1: 我试图通过将代码更改为...来使用MYSQL_ATTR_USE_BUFFERED_QUERY ...

$pdo = /* connection stuff here */
$pdo->setAttribute(PDO::MYSQL_ATTR_USE_BUFFERED_QUERY, true); // added code
$sql = "select 1; select 2; select 3; select 4;";
$statement = $pdo->query($sql);

do {
    $rowset = $statement->fetchAll();
    if($rowset) {               
        // Do stuff with $rowset
    }       
} while($statement->nextRowset());

这并没有解决问题

2 个答案:

答案 0 :(得分:3)

我认为你用do / while循环使事情变得复杂。

尝试使用简单的while循环:

$pdo = /* connection stuff here */
$sql = "select 1; select 2; select 3; select 4;";
$statement = $pdo->query($sql);

while($rowset = $statement->fetchAll()){
    //do stuff

    $statement->nextRowset();
}

这将继续循环,而rowset没有false值,然后该值应该与您期望的完全一致。

答案 1 :(得分:1)

为避免PDOException使用columnCount

while ($statment->columnCount()) {
     $rowset = $statment->fetchAll(PDO::FETCH_ASSOC);
     $statment->nextRowset();
}