就像 PDO 中的 PDO_Statment 对象一样,MySQLI_STMT 对象也是一个预处理语句所形成的对象,专门用来操作 MySQLi 所生成的预处理语句的。其实操作方式之类也都比较相似,不外乎以绑定参数为主的一些针对 SQL 语句和获取结果集的操作。

参数绑定及操作属性

之前的文章中想必大家已经见过我们使用的 bind_param() 方法,它与 PDO 中的 bindParam() 方法有很大的不同。

$stmt = $mysqli->prepare("insert into zyblog_test_user(username, password, salt) values(?, ?, ?)");

$username='mysqli_username';
$password='mysqli_password';
$salt = 'mysqli_salt';
$stmt->bind_param('sss', $username, $password, $salt); var_dump($stmt->insert_id); // int(232)
var_dump($stmt->affected_rows); // int(1) $stmt->execute(); $stmt->close();

首先就是之前提到过的,MySQLI_STMT 中绑定参数只能使用 ? 问号占位符,然后在使用 bind_param() 时,使用的是 's' 这种来按顺序绑定参数,这个 's' 代表的就是字符串。另外还可以是 'i' 表示整型数字、'd' 表示浮点数字、 'b' 表示 blob 类型。

另外,从上面的测试代码中还可以看出,可以使用一个 bind_param() 方法绑定多个参数,'sss' 就是三个字符串,按顺序进行绑定。

绑定参数之后,我们就可以通过 execute() 方法来执行语句。同 PDO 一样,这个方法只返回成功失败的信息,也就是一个布尔值。所以,我们需要通过 MySQLI_STMT 对象的 insert_id 来获得新增加数据的 ID ,或者通过 affected_rows 属性来获得当前语句执行后影响的行数,来确定语句是否真正地执行完成并达到我们的期望。

最后,我们使用 close() 关闭一个当前的 STMT 对象。这样在后面的操作中上面的 $stmt 对象就无法使用了。

接下来,我们看看如果绑定了错误的类型会怎么样,以及 MySQLI_STMT 中关于错误信息的提示。

$stmt = $mysqli->prepare("insert into zyblog_test_user(id, username, password, salt) values(?, ?, ?, ?)");

$id = 's';
$username='mysqli_username';
$password='mysqli_password';
$salt = 'mysqli_salt';
$stmt->bind_param('isss', $username, $password, $salt);
$stmt->execute(); var_dump($stmt->errno); // int(2031)
var_dump($stmt->error); // string(53) "No data supplied for parameters in prepared statement"
var_dump($stmt->error_list);
// array(1) {
// [0]=>
// array(3) {
// ["errno"]=>
// int(2031)
// ["sqlstate"]=>
// string(5) "HY000"
// ["error"]=>
// string(53) "No data supplied for parameters in prepared statement"
// }
// } $stmt->close();

在代码中,我们增加了 id 参数的绑定,然后指定的类型是 'i' ,但是,我们实际传递的变量是一个字符串类型,结果就会导致 MySQLI_STMT 产生错误。

可以看出,MySQLI_STMT 的错误属性和信息基本和 MySQLi 对象的是一样的。

列绑定

除了请求查询语句参数的绑定之外,MySQLI_STMT 也是支持直接绑定列的。就和 PDO 中的 bindColumn() 一样。

$stmt = $mysqli->prepare("select * from zyblog_test_user where username = ?");

$username = 'kkk';
$stmt->bind_param("s", $username); // 绑定参数
$stmt->bind_result($col1, $col2, $col3, $col4);
$stmt->execute(); // 执行语句 var_dump($stmt);
// object(mysqli_stmt)#2 (10) {
// ["affected_rows"]=>
// int(-1)
// ["insert_id"]=>
// int(0)
// ["num_rows"]=>
// int(0)
// ["param_count"]=>
// int(1)
// ["field_count"]=>
// int(4)
// ["errno"]=>
// int(0)
// ["error"]=>
// string(0) ""
// ["error_list"]=>
// array(0) {
// }
// ["sqlstate"]=>
// string(5) "00000"
// ["id"]=>
// int(3)
// } while($stmt->fetch()){
printf("%s %s %s %s", $col1, $col2, $col3, $col4);
echo PHP_EOL;
}
// 42 kkk 666 k6
// 43 kkk 666 k6
// …… var_dump($stmt->num_rows); // int(7) $stmt->close();

当然,方法的名称还是有些变动的。MySQLI_STMT 中绑定列的方法名为 bind_result() ,虽说名字不一样,但功能其实都是差不多的,查询语句中是几个列名,就要绑定几个列名。在这个表中,我们有四个字段,所以通过引用传递的方式绑定了 4 个列变量。当使用 fetch() 进行查询结果对象的遍历时,就像使用引用的方式为这 4 个列变量赋值。

在这段代码中,我们使用了 num_rows 这个属性来获得查询结果的行数量,这个属性是只针对 SELECT 语句的。上面介绍过的 affected_rows 是受影响的行数,这两个属性不是相同的概念哦!

返回结果集

执行 fetch() 方法返回的是一个布尔值,它主要的作用是将结果集绑定到指定的变量中,所以如果你直接打印它的结果是不会有什么有用的信息的,我们必须通过绑定列变量的方式来获得数据。而真正获得结果集的数据其实是通过另一个方法来获得一个 MySQLI_result 对象,然后再使用这个对象里面的方法就可以像 PDO 的 fetch() 一样来获得真正的结果集了。

$stmt = $mysqli->prepare("select * from zyblog_test_user where username = 'kkk'");

$stmt->execute(); // 执行语句
$result = $stmt->get_result(); while($row = $result->fetch_assoc()){
var_dump($row);
}
// array(4) {
// ["id"]=>
// int(42)
// ["username"]=>
// string(3) "kkk"
// ["password"]=>
// string(3) "666"
// ["salt"]=>
// string(2) "k6"
// }
// …… $stmt->close();

在这里,我们通过 get_result() 方法获得了一个结果集的 MySQLI_result 对象。然后通过该对象的 fetch_assoc() 就获得了键名形式的结果集数组。

关于 MySQLI_result 对象的内容,我们将在下篇文章中再进行详细的学习了解。

保存结果集及游标移动

最后就是关于游标的移动,上面的测试数据中我们可以查询到 7 条数据,并且第一条数据的 id 是 42 ,通过游标,我们可以不在 SQL 语句中使用 limit 而直接操作结果集来获取需要的数据。

$stmt = $mysqli->prepare("select * from zyblog_test_user where username = 'kkk'");

$stmt->bind_result($col1, $col2, $col3, $col4);
$stmt->execute(); // 执行语句
$stmt->store_result();
// 一共7条,从第5个开始
$stmt->data_seek(5);
$stmt->fetch();
printf("%s %s %s %s", $col1, $col2, $col3, $col4); // 47 kkk 666 k6
echo PHP_EOL; $stmt->close();

首先,我们需要使用 store_result() 方法来将结果集保存到内存中,这个方法和 MySQLi 中的 store_result() 方法是一样的。然后,通过 data_seek() 方法将游标移动 5 个位置,最后输出的结果就是后面那两条数据的内容了。是不是很高大上的感觉!

总结

关于 MySQLI_STMT 对象的内容还有一些,不过就不是那么常用了。从我们讲解的这些内容也可以看出它和 PDO 的许多不同之处。当然,总体的大方向基本还是一致的,所以我们学习起来也并不会有太大的困难,掌握理解之后更多的还是要动手操作,基本功可千万不能荒废哦!

测试代码:

https://github.com/zhangyue0503/dev-blog/blob/master/php/202009/source/8.PHP中的MySQLi扩展学习(五)MySQLI_STMT对象操作.php

参考文档:

https://www.php.net/manual/zh/book.mysqli.php

关注公众号:【硬核项目经理】获取最新文章

添加微信/QQ好友:【xiaoyuezigonggong/149844827】免费得PHP、项目管理学习资料

知乎、公众号、抖音、头条搜索【硬核项目经理】

B站ID:482780532

PHP中的MySQLi扩展学习(五)MySQLI_STMT对象操作的更多相关文章

  1. PHP中的MySQLi扩展学习(六)MySQLI_result对象操作

    在之前的文章中,我们就已经接触过 MYSQLI_result 相关的内容.它的作用其实就是一个查询的结果集.不过在 PDO 中,一般直接通过 query() 或者 PDOStatement 对象进行查 ...

  2. PHP中的MySQLi扩展学习(四)mysqli的事务与预处理语句

    对于 MySQLi 来说,事务和预处理语句当然是它之所以能够淘汰 MySQL(原始) 扩展的资本.我们之前也已经学习过了 PDO 中关于事务和预处理语句相关的内容.所以在这里,我们就不再多讲理论方面的 ...

  3. PHP中的MySQLi扩展学习(三)mysqli的基本操作

    我们继续 MySQLi 扩展的学习,上篇文章中提到过,MySQLi 的扩展相对于 PDO 来说功能更加的丰富,所以我们依然还会在学习过程中穿插各种 MySQLi 中好玩的方法函数.不过,今天的主角是 ...

  4. PHP中的MySQLi扩展学习(二)mysqli类的一些少见的属性方法

    虽说是少见的一些属性方法,但是可能还是有不少同学在日常的开发中使用过,这里只是学习了可能相对来说我们用得比较少的一些 mysqli 的属性或方法.就当是扩展一下自己的知识体系. 切换用户 首先就是切换 ...

  5. PHP中的MySQLi扩展学习(一)MySQLi介绍

    关于 PDO 的学习我们告一段落,从这篇文章开始,我们继续学习另外一个 MySQL 扩展,也就是除了 PDO 之外的最核心的 MySQLi 扩展.可以说它的祖先,也就是 MySQL(原始) 扩展是我们 ...

  6. 前端学PHP之面向对象系列第五篇——对象操作

    × 目录 [1]对象克隆 [2]对象比较[3]对象串行化[4]json 前面的话 本文主要介绍面向对象中的一些对象操作 对象克隆 对象复制,又叫对象克隆,可以通过 clone 关键字来完成 在多数情况 ...

  7. 解决phpMyAdmin中缺少mysqli扩展的错误

  8. phpmyadmin中缺少mysqli扩展 的结解办法

    修改 ;extension=php_mysqli.dll  去掉前面的 ;     以及 调整 php文件夹的目录位置.     这个办法是不是好使,我不确定.这个方法只适合 用win系统 这个,貌似 ...

  9. php学习五:数组操作

    前言:由于之前接触过js,所以在学习php的时候,里面继承了js的许多方法,所以数组里面的许多操作和js很相似,但是却比js里面多的多,个人感觉php是集成数组方法最多的一个 ,今天学习的有二十余个, ...

随机推荐

  1. 线程的分离状态(detached state)

    说到线程的分离状态,我认为,之所以会有这个状态,是因为系统对某些线程的终止状态根本不感兴趣导致的. 我们知道,进程中的线程可以调用: [cpp] view plaincopyprint? int pt ...

  2. WPF下获取文件运行路径、运行文件名等

    在客户端开发过程中,经常需要获取相对路径的一些资源,而相对路径的就与客户端运行文件的路径息息相关了.在以前的winform开发中,我们可以使用 System.Windows.Forms.Applica ...

  3. NPM使用方法

    什么是npm npm是nodejs的包管理器,在当今工程化前端开发过程中,npm包起着举足轻重的作用. 安装npm 作为nodejs的包管理器,npm随着nodejs一起安装的.通常情况下,当我们安装 ...

  4. Java线程池中submit()和execute()方法有什么区别

    两个方法都可以向线程池提交任务,execute()方法的返回类型是void,它定义在Executor接口中,而submit()方法返回有计算结构的Future对象,它定义在ExecutorServic ...

  5. 【spring】spring 核心注解

    注解具体分类如下: 1.模式注解 @Repository             数据仓储模式注解 @Component            通用组件模式注解 @Service            ...

  6. java 将字符串拆分成块装数组

    split 将字符串拆分 regex=???,根据???以其为界进行拆分. public String[] split(String regex) 根据给定正则表达式的匹配拆分此字符串. 该方法的作用 ...

  7. 使用spring向service里面注入dao不成功。

    因为原来的程序没有使用spring.后来加spring的时候action有个地方的new没有改!!! new了个新的实现层 不是spring管理的对象.

  8. 整理之BroadcaseReceiver

    广播的分类 有序广播:按接收器优先级从高到低接受消息,一次只能有一个接收器处理消息.中途可以被截断. 无序广播:所有接收器同时接受消息并处理,无法拦截. 本地广播:只能在本应用内传播的无需广播.上面两 ...

  9. leaflet加载离线OSM(OpenStreetMap)

    本文为博主原创,如需转载需要署名出处. leaflet作为广为应用的开源地图操作的API,是非常受欢迎,轻量级的代码让使用者更容易操作. 废话不多说,下面直接给出范例. 首先在这个网站下载leafle ...

  10. Pulsar の 保证消息的顺序性、幂等性和可靠性

    原文链接:Pulsar の 保证消息的顺序性.幂等性和可靠性 一.背景 前面两篇文章,已经介绍了关于Pulsar消费者的详细使用和自研的Pulsar组件. 接下来,将简单分析如何保证消息的顺序性.幂等 ...