【PHP】当mysql遇上PHP

本文涉及的产品
RDS MySQL Serverless 基础系列,0.5-2RCU 50GB
云数据库 RDS MySQL,高可用系列 2核4GB
RDS MySQL Serverless 高可用系列,价值2615元额度,1个月
简介: 博客提纲利用PHP连接mySQL数据库两套接口:面向对象和面向过程实现写改删查(CUBD)实例通过prepare语句处理相同类型的不同SQL语句通过bind_param()绑定参数,及相关注意事项通过bind_result()绑定结果,以及相关注意事项将文本写入数据库前应做的检测和处理检查是否为空(未输入值)去首尾空格魔鬼字符串转义一.利用PHP连接mySQL数据库这要从一个故事说起。

博客提纲

  • 利用PHP连接mySQL数据库
    • 两套接口:面向对象和面向过程
    • 实现写改删查(CUBD)实例
  • 通过prepare语句处理相同类型的不同SQL语句
    • 通过bind_param()绑定参数,及相关注意事项
    • 通过bind_result()绑定结果,以及相关注意事项
  • 将文本写入数据库前应做的检测和处理
    • 检查是否为空(未输入值)
    • 去首尾空格
    • 魔鬼字符串转义

一.利用PHP连接mySQL数据库

这要从一个故事说起。
某一天,一位名叫MySQL的农夫的一把斧子(数据库操作)掉进了一条名为PHP的河里,这时候,一位好心的河神出现了

PHP河的河神问他。。。。

下面,咱们还是说正经的把!。。。(:3 」∠)

在我主机(localhost)penghuwan数据库下,有张mytable的表如下图所示

1

PHP针对mysql数据库的操作有两套接口:面向对象接口和面向过程接口

  • 面向对象接口:通过调用对象中的函数完成数据库操作
  • 面向过程接口:直接调用PHP内置的函数实现数据库操作

因为执行写改删操作的PHP语句类似,所以这里只以“写操作”和“查操作”为例子

读操作:

面向对象:

<?php
   @$mysqli =  new mysqli('localhost', 'root', 'phw441423', 'penghuwan');//(主机,账号,密码,数据库) 返回一个mysqli对象
   if($mysqli->connect_error){//当有连接错误的时候,结束脚本运行并且报错
     die('连接错误,这个错误是'.$mysqli->connect_error);//die()函数:1结束脚本运行,2输出一段文本(括号内)
   }
   $query = "SELECT * FROM mytable";//把一段SQL语句保存在$query变量中
   $mysqli_result = $mysqli->query($query);//通过调用上面返回的mysqli对象中的方法,返回一个结果集对象(mysqli_result)
   while($row = $mysqli_result->fetch_assoc()){//调用mysqli_result的方法fetch_assoc()后,返回的是一个数组变量$row
     echo $row['name'];//访问返回数组变量$row中的数组成员,对应mytable表中的name列
     echo $row['number'];;//访问返回数组变量$row中的数组成员,对应mytable表中的number
     echo "<br/>";
   }
   $mysqli_result->free();//释放结果集
   $mysqli ->close();//关闭数据库连接
?>

首先通过

new mysqli($host, $username, $passwd, $dbname)

获取一个mysqli对象,然后在下面我们就可以通过调用对象中的方法query方法去实现写改删查

运行结果

思维导图

上面的例子中,一个关键的方法是mysqli对象query方法,意为查询.但实际上,它除了能运行“查”的SQL语句外,还能运行“写改删”的SQL语句。

关于query的返回值:

  • 执行失败,返回false
  • 执行成功
  • 如果执行的语句,即query是SELECT,SHOW,EXPLAIN 或 DESCRIBE,则返回一个结果集对象
  • 如果是其他,则返回false

面向过程:

<?php
   @$mysqli = mysqli_connect('localhost', 'root', 'phw441423', 'penghuwan');//(主机,账号,密码,数据库) 返回一个mysqli对象
   if(mysqli_connect_error()){//当有连接错误的时候,结束脚本运行并且报错
       die('连接错误,这个错误是'.mysqli_connect_error());;//die()函数:1结束脚本运行,2输出一段文本(括号内)
   }
   $query = "SELECT * FROM mytable";//把一段SQL语句保存在$query变量中
   $mysqli_result = mysqli_query($mysqli, $query);//在面向过程风格里,$mysqli对象成了该方法中的参数,也返回一个结果集对象(mysqli_result)
   while($row = mysqli_fetch_assoc($mysqli_result)){// 返回的是一个数组变量$row
     echo $row['name'];//访问返回数组变量$row中的数组成员,对应mytable表中的name列
     echo $row['number'];;//访问返回数组变量$row中的数组成员,对应mytable表中的number
     echo "<br/>";
   }
   mysqli_free_result($mysqli_result);//释放结果集
   mysqli_close($mysqli);//关闭数据库连接
?>

【注意点】

  1. mysqli_fetch_assoc(面向过程)和fetch_assoc(面向对象)这两个方法返回的是一个关联数组变量$row
  2. 在命令行界面里,我们需要做选择数据库的选择,即使用“USE 所选数据库”这个命令,但在这里我们在一开始连接的时候就选择了数据库了。例如:mysqli_connect('localhost', 'root', 'phw441423', 'penghuwan');中我们选择了数据库penghuwan所以就不用写USE语句了
  3. 最后记得要释放结果集和关闭连接

拥有两套接口固然增加了记忆难度,但如果你注意观察的话,两套接口函数的名称是联系紧密的。

  • 如何记忆?
    一般情况下:面向过程函数名= mysqli_ +面向对象函数名
    例如:
    返回结果集对象的方法:
    面向对象:query 面向过程:mysqli_query
    从结果集对象中返回某一行(形式为关联数组)的方法:
    面向对象:fetch_assoc 面向过程:mysqli_fetch_assoc

  • 两者联系
    一般情况下,面向对象接口中的对象将会成为面向过程接口中的第一个参数
    例如:
    通过mysqli对象取得结果集的时候:
    面向过程:$mysqli_result = mysqli_query($mysqli, $query);
    面向对象:$mysqli_result = $mysqli->query($query);


写操作:

面向对象:

<?php
   @$mysqli =  new mysqli('localhost', 'root', 'phw441423', 'penghuwan');//(主机,账号,密码,数据库) 返回一个mysqli对象
   if($mysqli->connect_error){//当有连接错误的时候,结束脚本运行并且报错
     die('连接错误,这个错误是'.$mysqli->connect_error);//die()函数:1结束脚本运行,2输出一段文本(括号内)
   }
   $query = "INSERT INTO mytable VALUES('C',30)";//把一段SQL语句保存在$query变量中
   $mysqli->query($query);// 此时返回的不是结果集对象,而是一个boolean,代表成功或失败
   $mysqli ->close();//关闭数据库连接
?>

运行结果:

3

思维导图

面向过程:

和第一个“查”的例子类似,这里不多加赘述。

二.通过prepare语句处理相同类型的不同SQL语句

通过bind_param()绑定参数,及相关注意事项

在实际操作中,我们可能需要处理大量相同类型的不同SQL语句,例如

"SELECT * FROM mytable WHERE name = ‘A’"

或者

"SELECT * FROM mytable WHERE name = ‘B’"

这样的语句。你可能会试图自己封装函数来避免写一大堆相同类型的语句。但实际上,PHP已经给我们封装好了一系列的内置函数,它就是prepare语句

我们接下来实现这样一段PHP脚本
通过prepare语句给mytable插入两行数据(类型相同的不同SQL语句)

我们原来的mytable表长这样

4

我们下面向其中插入两行

列1 列2
D 40
E 50
<?php
   @$mysqli = new mysqli('localhost', 'root', 'phw441423', 'penghuwan');//(主机,账号,密码,数据库) 返回一个mysqli对象
   if($mysqli->connect_error){//当有连接错误的时候,结束脚本运行并且报错
     die('连接错误,这个错误是'.$mysqli->connect_error);//die()函数:1结束脚本运行,2输出一段文本(括号内)
   }
   $query = "INSERT INTO mytable VALUES(?,?)";//,“?”表示模板中要被实际替换的变量
   $stmt = $mysqli->prepare($query);//通过prepare函数生成mysqli_statement对象
 
   $name1 = 'D';
   $number1 = 40;
   $stmt->bind_param("si",$name1,$number1);//通过mysqli_statement对象的bind_param方法用实际的变量替换模板中的"?"
   $stmt->execute();//第一次执行
 
   $name2 = 'E';
   $number2 = 50;
   $stmt->bind_param("si",$name2,$number2);//通过mysqli_statement对象的bind_param方法用实际的变量替换模板中的"?"
   $stmt->execute();//第二次执行
 
   $stmt->close();//关闭mysqli_statement
   $mysqli ->close();//关闭数据库连接
 
?>

思维导图

运行结束后

5

关键的一个方法是bind_param()方法,它接受多个参数,其中
第一个参数代表后面参数的类型。

第一个参数是一个字符串,由固定顺序的字符组成,这些字符包括“s”,”i”,”d”,”b”,分别表示字符串,整型,双精度和二进制文本,依次代表后面参数的类型。

字符 代表类型
“s” 字符串
“i” 整型
“d” 双精度
“b” 二进制文本

例如:我们上面的$stmt->bind_param("si",$name1,$number1);代表:$name1是字符串类型,,$number1是整型

【注意】

  • 不能直接向bind_param()第二个即以后的参数中写入具体的变量值!否则会报错:
    例如,我们把:
    <?PHP
      $name1 = 'D';
      $number1 = 40;
      $stmt->bind_param("si",$name1,$number1);
    ?>

改成:

<?PHP
   $stmt->bind_param("si",'D',40);
?>

运行:

6
【注意】

  • 你只能写入变量的名称而不能写具体的类型值——
    一个bind_param()函数对应一个execute()函数,如果连续写多个bind_param()再写execute()函数,相当于最后一个bind_param()覆盖前面写的的 bind_param()

例如我们把上面的

 $name1 = 'D';
 $number1 = 40;
 $stmt->bind_param("si",$name1,$number1);//通过mysqli_statement对象的bind_param方法用实际的变量替换模板中的"?"
 $stmt->execute();//第一次执行
 
 $name2 = 'E';
 $number2 = 50;
 $stmt->bind_param("si",$name2,$number2);//通过mysqli_statement对象的bind_param方法用实际的变量替换模板中的"?"
 $stmt->execute();//第二次执行
 

改成:

 $name1 = 'D';
 $number1 = 40;
 $stmt->bind_param("si",$name1,$number1);//通过mysqli_statement对象的bind_param方法用实际的变量替换模板中的"?"
 
 $name2 = 'E';
 $number2 = 50;
 $stmt->bind_param("si",$name2,$number2);//通过mysqli_statement对象的bind_param方法用实际的变量替换模板中的"?"
 $stmt->execute();//第二次执行
 

运行结果:

7

它并不会批量执行$name1,$number1和$name2,$number2的插入,而是只插入了$name2,$number2,因为最后一个bind_param()覆盖前面写的的 bind_param()

通过bind_result()绑定结果,及相关注意事项

上面的例子中我们演示了如何绑定参数,下面我来演示如何绑定结果,这里将用到bind_result()函数:

<?php
   @$mysqli = new mysqli('localhost', 'root', 'phw441423', 'penghuwan');//(主机,账号,密码,数据库) 返回一个mysqli对象
   if($mysqli->connect_error){//当有连接错误的时候,结束脚本运行并且报错
     die('连接错误,这个错误是'.$mysqli->connect_error);//die()函数:1结束脚本运行,2输出一段文本(括号内)
   }
   $query = "SELECT * FROM mytable";//prepare函数的参数
   $stmt = $mysqli->prepare($query);//通过prepare函数生成mysqli_statement对象
   $stmt->bind_result($name,$number);//将执行结果绑定到$name和,$number中
   $stmt->execute();// 执行生成查询结果
   while($stmt->fetch()){// 将查询结果中的第一行的列值分别赋给$name和,$number,同时游标移到下一行
    echo $name.'   '.$number;//输出mytable中当前行各个列的列值
    echo "<br/>";
   }
   $stmt->close();//关闭mysqli_statement
   $mysqli ->close();//关闭数据库连接
?>

思维导图

运行结果如下:

8

【注意】

  • bind_param必须放在execute语句的前面,但bind_result放在execute前后均可
    例如:我们将上面对应的代码改成:
    $stmt->execute();// 执行生成查询结果
    $stmt->bind_result($name,$number);//将执行结果绑定到$name和,$number中

运行结果同上(但注意bind_result应放在fetch语句前)

  • excute()执行完毕的时候,$name,$number仍为空,直到fetch()第一次执行的时候,$name,$number才取到对应行的列值
    将上面例子中对应代码改成:
    $stmt->bind_result($name,$number);//将执行结果绑定到$name和,$number中
    $stmt->execute();// 执行生成查询结果
    echo 'execute执行后$name的值为';
    var_dump($name);

运行结果:

9

  • 同一个prepare模板可多次使用,但前后使用两个prepare模板中间,必须关闭现有的mysqli_statement
    $query1 = "SELECT name FROM mytable";
     $stmt = $mysqli->prepare($query1);
     $stmt->execute();//执行第一个prepare模板语句
     
     $query2 = "SELECT number FROM mytable";//prepare函数的参数
     $stmt = $mysqli->prepare($query2);
     $stmt->execute();//执行第二个prepare模板语句 [注],这就是24行

提示的错误是,我对一个boolean值调用了execute函数
我尝试输出$stmt(最下面那个),输出为false(这里不做展示了)
这说明执行第二个prepare模板语句的时候失败了,那这时候该怎么办呢?

让我们在两段prepare模板语句间加上

$stmt->close():

即:

 <?PHP
   $query1 = "SELECT name FROM mytable";
   $stmt = $mysqli->prepare($query1);
   $stmt->execute();//执行第一个prepare模板语句
 
   $stmt->close();
 
   $query2 = "SELECT number FROM mytable";//prepare函数的参数
   $stmt = $mysqli->prepare($query2);
   $stmt->execute();//执行第二个prepare模板语句
?>

运行:报错消失

三.将字符串写入数据库前应做的检测和处理

应该注意的是三个方面的事情:

  1. 检查输入是否为空值,这点就不加赘述了
  2. 去除首尾空格(假设我们在录入数据库前没有去除空格的话,例如将“【空格】彭湖湾”录入数据库,那么在进行“【空格】彭湖湾”===“彭湖湾”的匹配时便会返回false)
  3. 对魔术字符串转义(如果不进行转义,字符串中的双引号和单引号会对我们的SQL语句造成干扰)
    <?php
      $text = $_GET['text'];// 从from表单中name属性为“text”的输入框中取得值 
      if(!$text){//如果text为空则输出警告,并结束脚本
         echo '您还没有输入任何值哦';
         exit();
      }
      $text = trim($text);//去除首尾空格
        if(!get_magic_quotes_gpc()){//检查是否自动开启了魔术字符串转义,如果没有,则手动转义魔术字符串
        $text = addslashes($text);
      }
      echo '经过处理后的值'.$text;
      echo "<br/>";
      echo '重新取出值'.stripslashes($text);
    ?>

输入空值的时候:

10

输入带空格和魔术字符串的文本——“【空格】penghuwan”

11

参考资料
《php和mysql的web开发》--(澳)威利,(澳)汤姆森 著
PHP官方文档 链接:http://php.net/manual/zh/
stackOverFlow社区 链接: https://stackoverflow.com/

其实啊,我只是把你们喝咖啡的时间,都用来喝啤酒而已
相关实践学习
每个IT人都想学的“Web应用上云经典架构”实战
本实验从Web应用上云这个最基本的、最普遍的需求出发,帮助IT从业者们通过“阿里云Web应用上云解决方案”,了解一个企业级Web应用上云的常见架构,了解如何构建一个高可用、可扩展的企业级应用架构。
MySQL数据库入门学习
本课程通过最流行的开源数据库MySQL带你了解数据库的世界。 &nbsp; 相关的阿里云产品:云数据库RDS MySQL 版 阿里云关系型数据库RDS(Relational Database Service)是一种稳定可靠、可弹性伸缩的在线数据库服务,提供容灾、备份、恢复、迁移等方面的全套解决方案,彻底解决数据库运维的烦恼。 了解产品详情:&nbsp;https://www.aliyun.com/product/rds/mysql&nbsp;
目录
相关文章
|
5月前
|
关系型数据库 MySQL Linux
查看Linux、Apache、MySQL、PHP版本的技巧
以上就是查看Linux、Apache、MySQL、PHP版本信息的方法。希望这些信息能帮助你更好地理解和使用你的LAMP技术栈。
269 17
|
6月前
|
关系型数据库 MySQL PHP
源码编译安装LAMP(HTTP服务,MYSQL ,PHP,以及bbs论坛)
通过以上步骤,你可以成功地在一台Linux服务器上从源码编译并安装LAMP环境,并配置一个BBS论坛(Discuz!)。这些步骤涵盖了从安装依赖、下载源代码、配置编译到安装完成的所有细节。每个命令的解释确保了过程的透明度,使即使是非专业人士也能够理解整个流程。
133 18
|
7月前
|
关系型数据库 MySQL 网络安全
如何排查和解决PHP连接数据库MYSQL失败写锁的问题
通过本文的介绍,您可以系统地了解如何排查和解决PHP连接MySQL数据库失败及写锁问题。通过检查配置、确保服务启动、调整防火墙设置和用户权限,以及识别和解决长时间运行的事务和死锁问题,可以有效地保障应用的稳定运行。
307 25
|
9月前
|
存储 关系型数据库 MySQL
PHP与MySQL动态网站开发:从基础到实践####
本文将深入探讨PHP与MySQL的结合使用,展示如何构建一个动态网站。通过一系列实例和代码片段,我们将逐步了解数据库连接、数据操作、用户输入处理及安全防护等关键技术点。无论您是初学者还是有经验的开发者,都能从中获益匪浅。 ####
|
9月前
|
关系型数据库 MySQL PHP
php实现一个简单的MySQL分页
通过本文的详细步骤和代码示例,我们实现了一个简单的PHP MySQL分页功能。主要步骤包括计算总记录数、设置分页参数、查询当前页的数据以及生成分页链接。这种分页方式适用于大多数Web应用,能够有效提升用户体验和页面响应速度。
244 4
|
9月前
|
SQL 关系型数据库 MySQL
PHP与MySQL的高效交互:从基础到实践####
本文深入探讨了PHP与MySQL数据库之间的高效交互技术,涵盖了从基础连接到高级查询优化的全过程。不同于传统的摘要概述,这里我们直接以一段精简代码示例作为引子,展示如何在PHP中实现与MySQL的快速连接与简单查询,随后文章将围绕这一核心,逐步展开详细讲解,旨在为读者提供一个从入门到精通的实战指南。 ```php <?php // 数据库配置信息 $servername = "localhost"; $username = "root"; $password = "password"; $dbname = "test_db"; // 创建连接 $conn = new mysqli($se
212 0
|
10月前
|
关系型数据库 MySQL PHP
PHP与MySQL的深度整合:构建高效动态网站####
在当今这个数据驱动的时代,掌握如何高效地从数据库中检索和操作数据是至关重要的。本文将深入探讨PHP与MySQL的深度整合方法,揭示它们如何协同工作以优化数据处理流程,提升网站性能和用户体验。我们将通过实例分析、技巧分享和最佳实践指导,帮助你构建出既高效又可靠的动态网站。无论你是初学者还是有经验的开发者,都能从中获得宝贵的见解和实用的技能。 ####
100 0
|
关系型数据库 MySQL 数据库连接
PHP 原生连接 Mysql
PHP 原生连接 Mysql
243 0
|
关系型数据库 MySQL PHP
【PHP 开发专栏】PHP 连接 MySQL 数据库的方法
【4月更文挑战第30天】本文介绍了 PHP 连接 MySQL 的两种主要方法:mysqli 和 PDO 扩展,包括连接、查询和处理结果的基本步骤。还讨论了连接参数设置、常见问题及解决方法,如连接失败、权限和字符集问题。此外,提到了高级技巧如使用连接池和缓存连接信息以优化性能。最后,通过实际案例分析了在用户登录系统和数据管理中的应用。
834 1
|
存储 SQL 前端开发
【PHP】一文详解如何连接Mysql数据库(附源码)
本文主要讲解PHP如何连接数据库并且根据前端的form表单提交的数据返回到数据库最后查询出来展现
452 0
【PHP】一文详解如何连接Mysql数据库(附源码)

推荐镜像

更多