为什么 affected_rows 总是返回 -1?

10
我好像在执行INSERT和SELECT时无法获取到affected_rows的值,它总是返回-1,我使用的是一个数据库类,在我的项目中一直使用这个类,并且该类使用MYSQLI准备语句来避免SQL注入。
有人知道为什么它总是返回-1吗?根据我的阅读,它应该能够返回插入和选择操作的受影响行数。
数据库类
class database {
    protected $_mysqli;
    protected $_debug;
 
    public function __construct($host, $username, $password, $database, $debug) {
        $this->_mysqli = new mysqli($host, $username, $password, $database);
        $this->_debug = (bool) $debug;
        if (mysqli_connect_errno()) {
            if ($this->_debug) {
                echo mysqli_connect_error();
                debug_print_backtrace();
            }
            return false;
        }
        return true;
    }
 
    public function q($query) {
        if ($query = $this->_mysqli->prepare($query)) {
            if (func_num_args() > 1) {
                $x = func_get_args();
                $args = array_merge(array(func_get_arg(1)),
                    array_slice($x, 2));
                $args_ref = array();
                foreach($args as $k => &$arg) {
                    $args_ref[$k] = &$arg; 
                }
                call_user_func_array(array($query, 'bind_param'), $args_ref);
            }
            $query->execute();
            
            if ($query->errno) {
              if ($this->_debug) {
                echo mysqli_error($this->_mysqli);
                debug_print_backtrace();
              }
              return false;
            }
 
            if ($query->affected_rows > -1) {
                return $query->affected_rows;
            }
            $params = array();
            $meta = $query->result_metadata();
            while ($field = $meta->fetch_field()) {
                $params[] = &$row[$field->name];
            }
            call_user_func_array(array($query, 'bind_result'), $params);
 
            $result = array();
            while ($query->fetch()) {
                $r = array();
                foreach ($row as $key => $val) {
                    $r[$key] = $val;
                }
                $result[] = $r;
            }
            $query->close(); 
            return $result;
        } else {
            if ($this->_debug) {
                echo $this->_mysqli->error;
                debug_print_backtrace();
            }
            return false;
        }
    }
 
    public function handle() {
        return $this->_mysqli;
    }
    
    public function last_insert_id()
    {
        return $this->_mysqli->insert_id;
    }

    public function found_rowss()
    {
        return $this->_mysqli->affected_rows;
    }
}
1个回答

6

对于 prepare 创建的 Select 语句,您应该使用 $query->num_rows()mysqli_stmt_num_rows($query)

在执行 "INSERT IGNORE" 时,Insert 语句可能会给您带来被压制的错误,这可能导致 $query->affected_rows() 中出现 -1。

php.net 上的评论(第二个链接)建议您使用 $query->sqlstate=="00000" 检查错误。


请查看php.net (manual/en/mysqli-stmt.affected-rows)

"此函数仅适用于更新表的查询。要从 SELECT 查询中获取行数,请改用 mysqli_stmt_num_rows()。"

以及php.net (manual/en/mysqli.affected-rows)

"检查 mysqli->affected_rows 是否等于 -1 并不是确定 "INSERT IGNORE" 语句成功的好方法。例如:在插入一些包含由用户提供的数据的行时,仅当它们与指定的唯一约束匹配时才忽略重复键错误,并导致 mysqli->affected_rows 返回 -1,即使插入了行。(在 MySQL 5.0.85 linux 和 php 5.2.9-2 windows 上进行了检查)。但是如果语句成功执行,则 mysqli->sqlstate 不返回错误。"


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接