博客提纲

  • 利用PHP连接mySQL数据库

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

一.利用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”

参考资料

《php和mysql的web开发》--(澳)威利,(澳)汤姆森 著

PHP官方文档 链接:http://php.net/manual/zh/

stackOverFlow社区 链接: https://stackoverflow.com/

【PHP】当mysql遇上PHP的更多相关文章

  1. 当mysql遇上PHP

    博客提纲 利用PHP连接mySQL数据库 两套接口:面向对象和面向过程 实现写改删查(CUBD)实例 通过prepare语句处理相同类型的不同SQL语句 通过bind_param()绑定参数,及相关注 ...

  2. Centos7 上安装mysql遇上的问题:mysql无法正常启动

    第一次在Centos上安装mysql遇到的一些问题. 第一步就遇到问题,安装mysql-server报错没有可用包.  [解决方法] 先要安装mysql # wget http://repo.mysq ...

  3. MVC遇上bootstrap后的ajax表单模型验证

    MVC遇上bootstrap后的ajax表单验证 使用bootstrap后他由他自带的样式has-error,想要使用它就会比较麻烦,往常使用jqueyr.validate的话只有使用他自己的样式了, ...

  4. 敏捷遇上UML-需求分析及软件设计最佳实践(郑州站 2014-6-7)

      邀请函: 尊敬的阁下:我们将在郑州为您奉献高端知识大餐,当敏捷遇上UML,会发生怎样的化学作用呢?首席专家张老师将会为您分享需求分析及软件设计方面的最佳实践,帮助您掌握敏捷.UML及两者相结合的实 ...

  5. 敏捷遇上UML—软创基地马年大会(广州站 2014-4-19)

        我们将在广州为您奉献高端知识大餐,当敏捷遇上UML,会发生怎样的化学作用呢?首席专家张老师将会为您分享需求分析及软件设计方面的最佳实践,帮助您掌握敏捷.UML及两者相结合的实战技巧. 时间:2 ...

  6. 敏捷遇上UML——软创基地马年大会(深圳站 2014-3-15)

    邀请函: 尊敬的阁下: 我们将在深圳为您奉献高端知识大餐,当敏捷遇上UML,会发生怎样的化学作用呢?首席专家张老师将会为您分享需求分析及软件设计方面的最佳实践,帮助您掌握敏捷.UML及两者相结合的实战 ...

  7. 初识genymotion安装遇上的VirtualBox问题

    想必做过Android开发的都讨厌那慢如蜗牛的 eclipse原生Android模拟器吧! 光是启动这个模拟器都得花上两三分钟,慢慢的用起来手机来调试,但那毕竟不是长久之计,也确实不方便,后来知道了g ...

  8. SQL SERVER 2008 R2 SP1更新时,遇上共享功能更新失败解决方案

    SQL SERVER 2008 R2 SP1更新时,遇上共享功能更新失败的问题,可作如下尝试: 更新失败后,在windows的[事件查看器→应用程序]中找到来源为MsiInstaller,事件ID为1 ...

  9. 当创业遇上O2O,新一批死亡名单,看完震惊了!

    当创业遇上O2O,故事就开始了,总投入1.6亿.半年开7家便利店.会员猛增至10万……2015半年过去后,很多故事在后面变成了一场创业“事故”,是模式错误还是烧钱过度?这些项目的失败能给国内创业者带来 ...

随机推荐

  1. 【UI测试】--合理性

  2. UD系统主定制界面

  3. OPNET仿真软件资料合集

    1. OPEN中国代理商业 http://www.credit-top.com/page/Default.asp?pageID=105

  4. 20155312 2016-2017-2《Java程序设计》课程总结

    20155312 2016-2017-2<Java程序设计>课程总结 每周作业链接汇总 预备作业1:你期望的师生关系是什么? 预备作业2:做中学learning by doing个人感想 ...

  5. tp5中代替tp3.2中的一些方法

    U方法 U方法是TP中的生成路由的内置方法,现在这个方法可以完全使用url方法替换 I方法 之前的TP有个I方法用来接收请求参数,目前可以使用input方法替代 C方法 c方法被config方法代替

  6. javaScript正则表达式的使用

    今天看了一个正则的写法,回想一下,对于正则都忘记得差不多了,称这个时间整理一下,收集了一些以前的资料和查看了一些别人的资料,做一个小小的总结,方便自己以后查看,也希望能帮助到大家!!   欢迎指正,欢 ...

  7. Codeforces Round #523 (Div. 2) E. Politics(最小费+思维建图)

    https://codeforces.com/contest/1061/problem/E 题意 有n个点(<=500),标记第i个点的代价a[i],然后分别在这n个点建两棵树,对于每颗树的每个 ...

  8. Interrouter Signals

    summary of traditional NoC interrouter signals summary of SMART interrouter signals flit_valid and f ...

  9. .net 根据图片网络地址获取图片二进制字节数据流

    根据html路径获取图片的字节 /// <summary> ///根据html路径获取图片的字节 /// </summary> /// <param name=" ...

  10. 1.9yield方法

    yield()方法的作用放弃当前的cpu资源,将他让给其他的任务去占用cpu的执行时间,但放弃的时间不确定,有可能刚放弃,马上又获得cpu时间片 测试 package com.cky.thread; ...