【问题标题】:Why is mysqli giving a "Commands out of sync" error?为什么 mysqli 给出“命令不同步”错误?
【发布时间】:2011-04-07 14:53:10
【问题描述】:

我正在尝试运行以下内容。

<?php

$db = mysqli_connect("localhost","user","pw") or die("Database error");
mysqli_select_db($db, "database");

$agtid = $_POST['level'];

$sql = sprintf("call agent_hier(%d)", $agtid);

$result = mysqli_query($db, $sql) or exit(mysqli_error($db));

if ($result) {
    echo "<table border='1'>
        <tr><th>id</th>
        <th>name</th>
        <th>parent_id</th>
        <th>parent_name</th>
        <th>level</th>
        <th>email</th></tr>";

    while ($row = mysqli_fetch_assoc($result)) 
    {
        $aid = $row["id"];
        $sql2 = "SELECT * FROM members WHERE MEMNO = '$aid'";
        $result2 = mysqli_query($db,$sql2) or exit(mysqli_error($db));

            while ($newArray = mysqli_fetch_array($result2)) {
                $fname = $newArray['FNAME'];
                $lname = $newArray['LNAME'];
                $mi = $newArray['MI'];  
                $address = $newArray['ADDRESS'];    
                $city = $newArray['CITY'];  
                $state = $newArray['STATE'];    
                $zip = $newArray['ZIP'];
                            $kdate = $newArray['KDATE'];
                $date = abs(strtotime(date('m/d/Y')) - strtotime(date($kdate))) / (60 * 60 * 24);
            }

        echo sprintf("<tr><td>%s</td><td>%s</td><td>%s</td><td>%s</td><td>%s</td><td>%s</td></tr>",
            $row["id"],$row["name"],
            $row["parent_id"],$row["parent_name"],
            $row["level"],$row["email"]);
    }

    echo "</table>";
}

mysqli_free_result($result);
mysqli_close($db);

?>

如果我从以下位置删除行:

  $aid = $row["agent_id"];

到....

  $date = abs(strtotime(date('m/d/Y')) - strtotime(date($kdate))) / (60 * 60 * 24);
  }

一切都会好起来的。如果没有,我会收到以下错误:

命令不同步;您现在无法运行此命令

在研究中,我认为这可能是由于多个 MySQLi 查询同时运行,其中使用 mysqli_multi_query 但对于 the guide 中的所有样本和一般数据似乎并不适用。

有什么想法吗?

【问题讨论】:

  • 代码中的$date = abs(strtotime(date('m/d/Y')) - strtotime(date($kdate))) / (60 * 60 * 24); } 行在哪里?你说如果你删除它,但我看不到它。

标签: php mysql mysqli


【解决方案1】:

您必须通过存储过程关闭先前的连接保持。 而不是每次都关闭连接,您可以简单地使用:

mysqli_next_result($conn);

【讨论】:

  • 这是正确的答案。
  • @Pacerier - 在每次查询之前总是调用mysqli_next_result 有什么缺点吗?除了轻微的气味;理想情况下,我会追踪导致连接处于不完整状态的原因 - 但随后我会在那里调用它,所以我想总是调用它,然后继续前进。
【解决方案2】:

MySQL 客户端不允许您执行新查询,其中仍有行要从正在进行的查询中获取。请参阅 MySQL 文档中关于常见错误的 Commands out of sync

您可以使用mysqli_store_result() 从外部查询中预取所有行。这将在 MySQL 客户端中缓冲它们,因此从服务器的角度来看,您的应用程序已经获取了完整的结果集。然后,即使在从现在缓冲的外部结果集中获取行的循环中,您也可以执行更多查询。

或者您 mysqli_result::fetch_all() 将完整的结果集作为 PHP 数组返回,然后您可以循环该数组。

调用存储过程是一种特殊情况,因为存储过程有可能返回多个结果集,每个结果集可能有自己的一组行。这就是为什么@a1ex07 的答案提到使用mysqli_multi_query() 并循环直到mysqli_next_result() 没有更多的结果集。这是满足 MySQL 协议所必需的,即使在您的情况下,您的存储过程只有一个结果集。


PS:顺便说一句,我看到您正在执行嵌套查询,因为您有表示层次结构的数据。您可能需要考虑以不同方式存储数据,以便更轻松地查询它。我做了一个名为Models for Hierarchical Data with SQL and PHP 的演讲。我还在我的书SQL Antipatterns: Avoiding the Pitfalls of Database Programming 的一章中介绍了这个主题。


下面是如何在 CodeIgnitor 3.0.3 中实现mysqli_next_result()

system/database/drivers/mysqli/mysqli_driver.php 的第 262 行更改

protected function _execute($sql)
{
    return $this->conn_id->query($this->_prep_query($sql));
}

到这里

protected function _execute($sql)
{
    $results = $this->conn_id->query($this->_prep_query($sql));
    @mysqli_next_result($this->conn_id); // Fix 'command out of sync' error
    return $results;
}

这是自 2.x 以来一直存在的问题。我刚刚更新到 3.x,不得不将这个 hack 复制到新版本。

【讨论】:

  • 感谢您的宝贵建议。我最终以在另一个 php 页面 (php.net/manual/en/mysqli.store-result.php) 上更有意义的方式解决了我自己的问题(到目前为止......)。顺便说一句,很棒的演示文稿-我需要给它一些实时的时间,但似乎很有价值。然而,乍一看 - 可能对我来说是个流浪汉,因为我使用的是 MySQL - 我相信类似的理论也适用
  • 很好的答案,帮助我解决了类似的问题。谢谢
  • 所以即使我们只有一个使用数据库对象的查询,我们仍然应该调用mysqli_next_result()?
  • 是的,除非确认没有更多结果集要处理,否则活动查询不会“清除”。
【解决方案3】:

如果您还对存储过程进行了几次调用,并遇到上述错误,我有一个解决方案,先生。韦恩.

恕我直言,在某个地方,CALLStored Procedure 实际上会破坏连接。所以你所要做的就是重置数据库连接句柄。

您甚至可以在配置文件中设置一个函数来为您执行此操作,您只需调用该函数一次,然后再将任何 queryCALL 设置为 Stored Procedure

我的实现是(忽略$app,因为我正在开发silex 框架,它就在那里,YMMV)

function flushhandle() {

    global $app;
    global $db_host;
    global $db_user;
    global $db_pass;
    global $db_name;
    $app['mysqlio']->close();
    $app['mysqlio'] = new mysqli($db_host, $db_user, $db_pass, $db_name);
    return $app['mysqlio'];
}

【讨论】:

  • “搞乱了连接”怎么办?
【解决方案4】:

只需调用这个函数:

$this->free_result();

function free_result() {
        while (mysqli_more_results($this->conn) && mysqli_next_result($this->conn)) {

            $dummyResult = mysqli_use_result($this->conn);

            if ($dummyResult instanceof mysqli_result) {
                mysqli_free_result($this->conn);
            }
        }
    }

【讨论】:

    【解决方案5】:

    简单地说, 在调用 mysqli_free_result 之后,您必须调用 mysqli_next_result($db) 。

    mysqli_free_result($result); mysqli_next_result($db) mysqli_close($db);

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 2015-10-31
      • 2015-12-21
      • 1970-01-01
      • 2012-09-26
      • 1970-01-01
      • 2013-01-11
      相关资源
      最近更新 更多