PHP PDO 多选查询始终删除最后一个行集

Posted

技术标签:

【中文标题】PHP PDO 多选查询始终删除最后一个行集【英文标题】:PHP PDO multiple select query consistently dropping last rowset 【发布时间】:2014-02-24 11:45:35 【问题描述】:

我遇到了使用 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 ... 例程基于可在http://us1.php.net/manual/en/pdostatement.nextrowset.phpnextRowset() 函数的 PHP 文档中找到的内容

最后调用 $statement->closeCursor() 似乎不起作用

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

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

谁能解释一下导致最后一个行集被删除的原因?谢谢!

版本:PHP 5.4.12MySQL 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());

这并没有解决问题

【问题讨论】:

您是否尝试过启用 PDO::MYSQL_ATTR_USE_BUFFERED_QUERY? 我刚刚做了 - 见编辑 1 您确定最后一个查询是以分号结束的吗? 它适用于我的 4 个行集。我尝试删除最后一个分号而没有任何效果。篡改ATTR_EMULATE_PREPARES 允许或不允许执行多个语句。我在 MAMP 上运行 php 5.4。 嘿,最后一个行集是否可能未正确获取?您的代码中是否存在未正确评估的条件会使脚本仅获取一行,或者根本不获取?这将导致 $rowset 为空,并且 $statement 无法使用并产生您的 HY000 错误。 【参考方案1】:

我认为你的 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 工作...但它会产生类似array (0,1,2,3)array(0,1,2,3)array(0,1,2,3)... 问题是,当您致电array[0] 时,所有三个arrays @ 987654327@他们的第一个元素所以你有三个index [0]s....你能知道如何分开3个arrays例如保存每个returnedarray在一个单独的variable... cz现在它们都合而为一了。 尝试使用 fetch 而不是 fetchall【参考方案2】:

要避免PDOException,只需使用columnCount

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

【讨论】:

以上是关于PHP PDO 多选查询始终删除最后一个行集的主要内容,如果未能解决你的问题,请参考以下文章

删除foreach最后一个逗号PHP [重复]

php和mysql pdo查询

如何在使用 PHP-PDO 记录查询时获取最后一个插入 ID

PHP操作PDO预处理以及事务

PHP操作PDO预处理以及事务

PHP使用id和pdo删除记录(初学者)[重复]