当mysql遇上PHP

news2025/1/4 19:49:18

一.利用PHP连接mySQL数据库

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

PHP河的河神问他。。。。

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

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

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();//关闭数据库连接
?>

复制

运行结果:

思维导图

面向过程:

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

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

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

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

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

复制

或者

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

复制

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

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

我们原来的mytable表长这样

我们下面向其中插入两行

列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();//关闭数据库连接
 
?>

复制

思维导图

运行结束后

关键的一个方法是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);
?>

复制

运行:

【注意】

  • 你只能写入变量的名称而不能写具体的类型值—— 一个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();//第二次执行

复制

运行结果:

它并不会批量执行$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();//关闭数据库连接
?>

复制

思维导图

运行结果如下:

【注意】

  • 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);

复制

运行结果:

  • 同一个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);
    ?>

复制

输入空值的时候:

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

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/674111.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Kubernetes API Server源码学习(三):KubeAPIServer、APIExtensionsServer、AggregatorServer

本文基于Kubernetes v1.22.4版本进行源码学习 9、KubeAPIServer GenericAPIServer提供了一些通用的功能&#xff0c;其他的Server基于GenericAPIServer进行拓展&#xff0c;代码量就会减少许多 KubeAPIServer负责处理Kubernetes内建资源的REST请求&#xff0c;比如Pod、Deploy…

C/C++系列系统学习目录

友情链接&#xff1a;专栏地址 文章目录 一、C语言篇二、C篇 编程规范&#xff1a;C/C语言编程规范 一、C语言篇 章节内容1.初识C语言【C语言篇】初识C语言2.C语言最基础入门【C语言篇】C语言最基础入门3.C语言的输入输出相关知识【C语言篇】C语言的输入/输出相关知识4.C语言…

【Pytest实战】Pytest+Allure+Jenkins自动化测试框架搭建

&#x1f604;作者简介&#xff1a; 小曾同学.com,一个致力于测试开发的博主⛽️&#xff0c;主要职责&#xff1a;测试开发、CI/CD 如果文章知识点有错误的地方&#xff0c;还请大家指正&#xff0c;让我们一起学习&#xff0c;一起进步。&#x1f60a; 座右铭&#xff1a;不想…

从零开始理解Linux中断架构(5)--EL跃迁与Linux用户/内核态

ARM64系统Reset时,PE进入最高的异常级别运行状态 1)Reset后最高异常级别可以选用任何一种运行状态 2)cold reset由输入信号配置,warm reset由RMR_ELx.AA64配置 1)内核态EL1迁移到EL0t 上一节我们提到需要仔细理解的图吗?eret这条特殊的异常返回指令。 我们期望…

《机器学习公式推导与代码实现》chapter14-CatBoost

《机器学习公式推导与代码实现》学习笔记&#xff0c;记录一下自己的学习过程&#xff0c;详细的内容请大家购买作者的书籍查阅。 CatBoost CatBoost是俄罗斯搜索引擎巨头Yandex于2017年开源的一款GBDT计算框架&#xff0c;因能够高效处理数据中的类别特征而取名为CatBoost(C…

一时兴起之matlab学习记录

是学习记录&#xff0c;会有错误的地方 安装的话看其他文章把 小操作 查看历史命令 在输入命令的地方&#xff0c;按下↑的方向键即可 变量 对大小写敏感若想以指定的类型存储就是 类型名(值),如 int16(4)&#xff0c;这个也可以强转变量名字有限制&#xff0c;键入namele…

10分钟搭建Stable Diffusion

前言 人工智能生成内容&#xff08;Artificial Intelligence Generated Content&#xff0c;简称 AIGC&#xff09;是当下最火的概念之一。AIGC 被认为是继专业生成内容&#xff08;Professional Generated Content, PGC&#xff09;和用户生成内容&#xff08;User Generated…

【MySQL】数据库基础 ③

上一章&#xff1a; 【MySQL】数据库基础 ② ✍临时表 说明&#xff1a; MySQL 临时表在我们需要保存一些临时数据时是非常有用的。临时表只在当前连接可见&#xff0c;…

局域网实验报告

计算机网络综合实训 实训报告一 所在院系 计算机与信息工程学院 学科专业名称 计算机科学与技术 导师及职称 柯宗武 教授 提交时间 2023.3.10 网络层实验报告 &#xff08;湖北师范大学计算机与信息工程学院 中国 黄石 435002&#xff09; 1 集线器与交换机的对比实验 1.1 背…

【Python】自动化构建项目结构样式

引言 在使用Python或者其它编程语言的项目时候&#xff0c;编写README.md 往往是不可或缺的&#xff1b; 而在README.md 中&#xff0c;关于项目结构的样式展示&#xff0c;这个是可选的。不展示也无伤大雅&#xff0c;但有展示的话&#xff0c;有以下优点&#xff1a; 提供…

第九章 总结及作业(4)【编译原理】

第九章 总结及作业&#xff08;4&#xff09;【编译原理】 前言推荐第九章 运行时存储空间组织9.1 目标程序运行时的活动9.1.1过程的活动9.1.2参数传递 9.2 运行时存储器的划分9.2.1运行时存储器的划分9.2.2 活动记录9.2.3 存储分配策咯 9.3 静态存储分配9.3.1数据区*9.3.2公用…

基于深度学习的人脸检测技术

用到环境 1、pycharm community edition 2022.3.2 2、Python 3.10 整篇内容都已上传至我的csdn资源中&#xff0c;想用的请移步。 流程 多任务级联卷积神经网络(Multi-task Cascaded Convolutional Networks, MTCNN)算法进行人脸检测 普通人脸检测 单人人脸检测 图1 单人人…

我最喜欢的编程语言是python,以及我的见解!!

这里写目录标题 我最喜欢的编程语言&#xff1a;1、我个人认为编程语言优劣的评选标准2、我对不同编程语言的优点与缺点的拙见**1. Java****2. Python****3. JavaScript****4. C语言&#xff1a;****5. C语言&#xff1a;** 3、对python编程语言未来发展的猜测和未来趋势 我最喜…

Vicuna-13B使用云服务器部署

Vicuna概述 Vicuna由一群主要来自加州大学伯克利分校的研究人员推出&#xff0c;仍然是熟悉的配方、熟悉的味道。Vicuna同样是基于Meta开源的LLaMA大模型微调而来&#xff0c;它的训练数据是来自ShareGPT上的7万多条数据&#xff08;ShareGPT一个分享ChatGPT对话的谷歌插件&am…

kerberos配置dolphinscheduler

kerberos配置dolphinscheduler 一、添加dolphin 用戶1.所有節點上執行如下命令&#xff1a; 二、DolphinScheduler集群模式部署1.集群规划2.前置准备工作3.解压DolphinScheduler安装包4. 创建元数据库及用户5. 配置一键部署脚本6 初始化数据库7.修改common配置文件8. 一键部署D…

华为、思科和瞻博网络三个厂商如何配置基本ACL和高级ACL?

今天给大家带来基本ACL和高级ACL的配置&#xff0c;主要会介绍三个厂商的配置&#xff1a; 其他厂商也可以参考&#xff0c;比如华三的可以参考华为的&#xff0c;锐捷的参考思科的。 1. 基本ACL配置 基本ACL&#xff08;Access Control List&#xff09;是一种简单的网络安全…

【Java高级语法】(十二)可变参数:Java中的“可变之美“,做好这些细节,你的程序强大又灵活~

Java高级语法详解之可变参数 &#x1f539; 前言1️⃣ 概念2️⃣ 优势和缺点3️⃣ 特征和应用场景3.1 特征3.2 应用场景 4️⃣ 使用和原理5️⃣ 使用技巧5.1 可变参数结合泛型5.2 使用元组或列表进行参数传递5.3 使用默认值5.4 缓存计算结果 6️⃣ 实战&#xff1a;构建动态日志…

【Vue3+Ts project】认识 Websocket 以及 socket.io 库

目录 Websocket socket.io Socket.iO 事件名总结&#xff1a; Socket.IO 方法总结 Websocket 作用&#xff1a; WebSocket 仍然提供实时的双向通信功能&#xff0c;使用Vue3 应用程序能够与服务器进行实时数据交换降低延迟和网络开销&#xff1a;相比传统的HTTP请求-响…

scratch lenet(7): C语言计算可学习参数数量和连接数量

scratch lenet(7): C语言计算可学习参数数量和连接数量 1. 目的 按照 LeNet-5 对应的原版论文 LeCun-98.pdf 的网络结构&#xff0c;算出符合原文数据的“网络每层可学习参数数量、连接数量”。 网络上很多人的 LeNet-5 实现仅仅是 “copy” 现有的别人的项目&#xff0c; 缺…

求2的N次幂(C++)解决高精度运算

​&#x1f47b;内容专栏&#xff1a;《C/C专栏》 &#x1f428;本文概括&#xff1a; 计算高精度的2的N次方数字。 &#x1f43c;本文作者&#xff1a;花 碟 &#x1f438;发布时间&#xff1a;2023.6.22 文章目录 ​前言求2的N次方&#xff0c;N ≤ 10000实现思路&#xff1a…