mysqli给“命令不同步”的错误 – 为什么?

我试图运行以下。

<?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但对于指南中的所有示例和一般数据似乎并不适用。

有任何想法吗?

MySQL客户端不允许在执行正在进行的查询中仍然存在行的情况下执行新的查询。 查看MySQL文档中常见错误中的命令不同步 。

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

或者你mysqli_result::fetch_all()它返回完整的结果集为一个PHP数组,然后你可以遍历该数组。

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


PS:顺便说一下,我看到你正在做嵌套查询,因为你有代表层次结构的数据。 您可能要考虑以不同方式存储数据,以便您可以更轻松地查询。 我做了一个关于这个标题为SQL和PHP分层数据模型的演示文稿。 在我的书“ SQL反模式:避免数据库编程的陷阱”一章中,我也会介绍这个主题。


这里是如何在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,不得不复制这个黑客到新版本。

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

mysqli_free_result($结果); mysqli_next_result($ db) mysqli_close($ db);

只需调用这个函数:

 $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); } } } 

您必须通过存储过程closures以前的连接。 而不是每次closures连接,您可以简单地使用:

 mysqli_next_result($conn); 

如果您也正在对存储过程进行一些调用,并面对上述错误,那么我为您提供了一个解决scheme, Wayne先生

恕我直言,在某个地方, CALL Stored Procedure实际上弄乱了连接。 所以你所要做的就是重置数据库连接句柄。

你甚至可以在你的configuration文件中为你做一个函数,你只需要调用这个函数一次,然后再进行任何queryCALL Stored Procedure

我的实现是(只是忽略$app因为我正在工作在那里,YMMV的silex框架)

 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']; }