我正在尝试运行以下内容。
<?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中的所有样本和一般数据似乎不适用。
有什么想法吗?
答案 0 :(得分:38)
MySQL客户端不允许您执行新查询,其中仍有要从正在进行的查询中提取的行。有关常见错误,请参阅MySQL文档中的Commands out of sync。
您可以使用mysqli_store_result()
预取外部查询中的所有行。这将在MySQL客户端中缓冲它们,因此从服务器的角度来看,您的应用程序已获取完整的结果集。然后,即使在从now-buffered外部结果集中获取行的循环中,您也可以执行更多查询。
或者你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
更改
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复制到新版本。
答案 1 :(得分:3)
简单地说, 在调用mysqli_free_result之后,你必须调用mysqli_next_result($ db)。
mysqli_free_result($结果); 的 mysqli_next_result($分贝)强> mysqli_close($分贝);
答案 2 :(得分:2)
只需调用此函数:
$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);
}
}
}
答案 3 :(得分:0)
您必须通过存储过程关闭先前的连接保持。 您不必每次都关闭连接,只需使用:
mysqli_next_result($conn);
答案 4 :(得分:-1)
如果您还要对存储过程进行一些调用,并且遇到上述错误,我会为您提供解决方案,先生。韦恩
恕我直言,在某个地方,CALL
到Stored Procedure
实际上弄乱了连接。所以你要做的就是重置数据库连接句柄。
你甚至可以在你的配置文件中有一个功能为你做这个,你所做的就是在对query
CALL
或Stored Procedure
做任何$app
或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'];
}
,因为我正在处理{{1}}框架,YMMV)
{{1}}