设为首页 收藏本站
查看: 1009|回复: 0

[经验分享] 【PHP】当mysql遇上PHP

[复制链接]

尚未签到

发表于 2017-12-12 11:00:44 | 显示全部楼层 |阅读模式
博客提纲


  • 利用PHP连接mySQL数据库

    • 两套接口:面向对象和面向过程
    • 实现写改删查(CUBD)实例

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

    • 通过bind_param()绑定参数,及相关注意事项
    • 通过bind_result()绑定结果,以及相关注意事项

  • 将文本写入数据库前应做的检测和处理

    • 检查是否为空(未输入值)
    • 去首尾空格
    • 魔鬼字符串转义

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

  PHP河的河神问他。。。。
DSC0001.png

  下面,咱们还是说正经的把!。。。(:3 」∠)
  在我主机(localhost)penghuwan数据库下,有张mytable的表如下图所示
DSC0002.png

  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 = &quot;SELECT * FROM mytable&quot;;//把一段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 &quot;<br/>&quot;;
  }
  $mysqli_result->free();//释放结果集
  $mysqli ->close();//关闭数据库连接
  
?>
  

  首先通过
  

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

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

  思维导图
DSC0004.png

  上面的例子中,一个关键的方法是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 = &quot;SELECT * FROM mytable&quot;;//把一段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 &quot;<br/>&quot;;
  }
  mysqli_free_result($mysqli_result);//释放结果集
  mysqli_close($mysqli);//关闭数据库连接
  
?>
  

  【注意点】


  • mysqli_fetch_assoc(面向过程)和fetch_assoc(面向对象)这两个方法返回的是一个关联数组变量$row
  • 在命令行界面里,我们需要做选择数据库的选择,即使用“USE 所选数据库”这个命令,但在这里我们在一开始连接的时候就选择了数据库了。例如:mysqli_connect('localhost', 'root', 'phw441423', 'penghuwan');中我们选择了数据库penghuwan所以就不用写USE语句了
  • 最后记得要释放结果集和关闭连接
  拥有两套接口固然增加了记忆难度,但如果你注意观察的话,两套接口函数的名称是联系紧密的。


  •   如何记忆?
      
    一般情况下:面向过程函数名= 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 = &quot;INSERT INTO mytable VALUES('C',30)&quot;;//把一段SQL语句保存在$query变量中
  $mysqli->query($query);// 此时返回的不是结果集对象,而是一个boolean,代表成功或失败
  $mysqli ->close();//关闭数据库连接
  
?>
  

  运行结果:
DSC0005.png

  思维导图
DSC0006.png


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

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

通过bind_param()绑定参数,及相关注意事项
  在实际操作中,我们可能需要处理大量相同类型的不同SQL语句,例如
  

&quot;SELECT * FROM mytable WHERE name = ‘A’&quot;  

  或者
  

&quot;SELECT * FROM mytable WHERE name = ‘B’&quot;  

  这样的语句。你可能会试图自己封装函数来避免写一大堆相同类型的语句。但实际上,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 = &quot;INSERT INTO mytable VALUES(?,?)&quot;;//,“?”表示模板中要被实际替换的变量
  $stmt = $mysqli->prepare($query);//通过prepare函数生成mysqli_statement对象
  $name1 = 'D';
  $number1 = 40;
  $stmt->bind_param(&quot;si&quot;,$name1,$number1);//通过mysqli_statement对象的bind_param方法用实际的变量替换模板中的&quot;?&quot;
  $stmt->execute();//第一次执行
  $name2 = 'E';
  $number2 = 50;
  $stmt->bind_param(&quot;si&quot;,$name2,$number2);//通过mysqli_statement对象的bind_param方法用实际的变量替换模板中的&quot;?&quot;
  $stmt->execute();//第二次执行
  $stmt->close();//关闭mysqli_statement
  $mysqli ->close();//关闭数据库连接
  
?>
  

  思维导图
DSC0007.png

  运行结束后
DSC0008.png

  关键的一个方法是bind_param()方法,它接受多个参数,其中
  
第一个参数代表后面参数的类型。
  第一个参数是一个字符串,由固定顺序的字符组成,这些字符包括“s”,”i”,”d”,”b”,分别表示字符串,整型,双精度和二进制文本,依次代表后面参数的类型。

字符
代表类型
“s”
字符串
“i”
整型
“d”
双精度
“b”
二进制文本  例如:我们上面的$stmt->bind_param(&quot;si&quot;,$name1,$number1);代表:$name1是字符串类型,,$number1是整型
  【注意】


  • 不能直接向bind_param()第二个即以后的参数中写入具体的变量值!否则会报错:  
    例如,我们把:

  

    <?PHP  $name1 = 'D';
  $number1 = 40;
  $stmt->bind_param(&quot;si&quot;,$name1,$number1);
  ?>
  

  改成:
  

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

  运行:
DSC0009.png

  
【注意】


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

  例如我们把上面的
  

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

  改成:
  

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

  运行结果:
DSC00010.png

  它并不会批量执行$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 = &quot;SELECT * FROM mytable&quot;;//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 &quot;<br/>&quot;;
  }
  $stmt->close();//关闭mysqli_statement
  $mysqli ->close();//关闭数据库连接
  
?>
  

  思维导图
DSC00011.png

  运行结果如下:
DSC00012.png

  【注意】


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

  运行结果:
DSC00013.png



  • 同一个prepare模板可多次使用,但前后使用两个prepare模板中间,必须关闭现有的mysqli_statement
  

    $query1 = &quot;SELECT name FROM mytable&quot;;  $stmt = $mysqli->prepare($query1);
  $stmt->execute();//执行第一个prepare模板语句
  $query2 = &quot;SELECT number FROM mytable&quot;;//prepare函数的参数
  $stmt = $mysqli->prepare($query2);
  $stmt->execute();//执行第二个prepare模板语句 [注],这就是24行
  

DSC00014.png

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

$stmt->close():  

  即:
  

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

  运行:报错消失

三.将字符串写入数据库前应做的检测和处理
  应该注意的是三个方面的事情:


  • 检查输入是否为空值,这点就不加赘述了
  • 去除首尾空格(假设我们在录入数据库前没有去除空格的话,例如将“【空格】彭湖湾”录入数据库,那么在进行“【空格】彭湖湾”===“彭湖湾”的匹配时便会返回false)
  • 对魔术字符串转义(如果不进行转义,字符串中的双引号和单引号会对我们的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 &quot;<br/>&quot;;
  echo '重新取出值'.stripslashes($text);
  ?>
  

  输入空值的时候:
DSC00015.png

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


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

DSC00017.jpg

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-423284-1-1.html 上篇帖子: MySQL字符集及校对规则的理解 下篇帖子: Open-Falcon 监控系统监控 MySQL/Redis/MongoDB 状态监控
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表