gpt4 book ai didi

php - 准备好的语句返回 False

转载 作者:行者123 更新时间:2023-11-29 06:06:41 25 4
gpt4 key购买 nike

我知道有很多问题与我的问题类似。但我真的想不通这里的问题。我有一个名为“UsersClass”的类,它负责与用户相关的每项任务。而我使用mysqli prepare语句从数据库中插入或选择数据,问题是很多情况下prepare语句返回false。我通过每次建立新连接解决了这个问题,但这导致了其他功能的另一个问题“2006:Mysql 服务器已经消失”请参见下面的代码:此处和其他函数中的 prepare 返回 false。

function isPostingAllowed() {
$this->setPsSelectUsers($this->_connection->prepare("Select * from userpermissions where userpermissions.UserId = ? and userpermissions.PermissionId = 1"));
$this->_psSelectUsers->bind_param('i',$this->UserId);
$this->_psSelectUsers->execute();
if ( false===$this->_psSelectUsers ) {
die('prepare() failed: ' . htmlspecialchars($this->_connection->error));}
if ($this->_psSelectUsers->fetch()){
return true;
}else{
return false;
}}


function setPsSelectUsers($stmt) {
$this->unsetPsSelectUsers();
// mysqli_close($this->_Connection);
// $this-> __construct();

$this->_psSelectUsers= $stmt;}

当我取消注释这两行时,第一个函数将起作用并且 prepare staement 不会返回 false,但在这种情况下,以下函数将抛出错误 2006:

 function checkUserAuthentication() {
$this->setPsSelectUsers($this->_connection->prepare("SELECT UserLogInName FROM Users WHERE UserLogInName=? AND UserPassword=?"));
$this->_psSelectUsers->bind_param('ss',$this->UserLogInName, $this->UserPassword);
$this->_psSelectUsers->execute();
if ($this->_psSelectUsers->fetch()){
return true;
}else{
return false;
}}

那么如何在不产生新问题的情况下解决第一个问题呢?

最佳答案

问题:

... I use mysqli prepare statement to insert or select data from the database, the problem is that in many cases prepare statement return false.

那是因为您以无序方式运行查询,即您在关闭前一个语句对象之前执行->prepare()。给定您当前的代码,在准备好的语句中添加以下错误报告代码,

if(!$this->_connection->prepare(...)){
printf('errno: %d, error: %s', $this->_connection->errno, $this->_connection->error);
die();
}

如果您查看 $this->_connection->error,您将看到以下错误,

Commands out of sync; you can't run this command now

此问题已在许多论坛中记录,例如:

  • 来自 MySQL documentation ,

    如果您的命令不同步;您现在无法在您的客户端代码中运行此命令,您正在以错误的顺序调用客户端函数。

  • 从这个SO thread ,

    您不能同时进行两个查询,因为 mysqli 默认使用无缓冲查询(对于准备好的语句;...

解决方案:

以正确的顺序执行命令,( See this example code )

  1. 打开连接(一次:在最开始,而不是在每个查询执行期间)
  2. 创建准备好的语句
  3. 绑定(bind)参数
  4. 执行查询
  5. 绑定(bind)结果变量
  6. 获取值到那些绑定(bind)变量
  7. 关闭语句对象
  8. 关闭连接(一次:在最后,而不是在每个查询执行期间)

(按照步骤 2 到 7 执行所有查询,但根据您的查询,一个或多个步骤可能是可选的)

所以解决方案是,在再次调用->prepare() 之前关闭之前的statement 对象。从 setPsSelectUsers() 方法中取出此方法调用 $this->unsetPsSelectUsers(); 并将其放在 if ($this->_psSelectUsers-> 之前fetch()){...}else{...} isPostingAllowed()checkUserAuthentication() 方法 block 。此外,将 $this->_psSelectUsers->fetch() 方法调用的状态保存在一个变量中,并在随后的 if block 中使用它。所以你的代码应该是这样的:

public function isPostingAllowed() {
$this->setPsSelectUsers($this->_connection->prepare("Select * from userpermissions where userpermissions.UserId = ? and userpermissions.PermissionId = 1"));
if(!$this->_psSelectUsers){
printf('errno: %d, error: %s', $this->_connection->errno, $this->_connection->error);
die();
}
$this->_psSelectUsers->bind_param('i',$this->UserId);
$this->_psSelectUsers->execute();
$status = $this->_psSelectUsers->fetch();
$this->unsetPsSelectUsers();
if ($status){
return true;
}else{
return false;
}
}

private function setPsSelectUsers($stmt){
$this->_psSelectUsers= $stmt;
}

private function unsetPsSelectUsers() {
if (isset($this->_psSelectUsers)) {
$this->_psSelectUsers->close();
unset($this->_psSelectUsers);
}
}

public function checkUserAuthentication() {
$this->setPsSelectUsers($this->_connection->prepare("SELECT UserLogInName FROM Users WHERE UserLogInName=? AND UserPassword=?"));
if(!$this->_psSelectUsers){
printf('errno: %d, error: %s', $this->_connection->errno, $this->_connection->error);
die();
}
$this->_psSelectUsers->bind_param('ss',$this->UserLogInName, $this->UserPassword);
$this->_psSelectUsers->execute();
$status = $this->_psSelectUsers->fetch();
$this->unsetPsSelectUsers();
if ($status){
return true;
}else{
return false;
}
}

此外,您不必再使用此setPsSelectUsers() 方法,您可以直接在您的方法中使用属性$_psSelectUsers,如下所示:

$this->_psSelectUsers = $this->_connection->prepare(...);

关于php - 准备好的语句返回 False,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/41080057/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com