php中防止SQL注入的最佳解决方法


Posted in PHP onApril 25, 2013

如果用户输入的是直接插入到一个SQL语句中的查询,应用程序会很容易受到SQL注入,例如下面的例子:

$unsafe_variable = $_POST['user_input'];
mysql_query("INSERT INTO table (column) VALUES ('" . $unsafe_variable . "')");

这是因为用户可以输入类似VALUE"); DROP TABLE表; - ,使查询变成:
INSERT INTO table (column) VALUES('VALUE'); DROP TABLE table;'

我们应该怎么防止这种情况呢?下面我们来看看Theo的回答

使用预备义语句和参数化查询。对于带有任何参数的sql语句都会被发送到数据库服务器,并被解析!对于攻击者想要恶意注入sql是不可能的!

实现这一目标基本上有两种选择:

1.使用PDO(PHP Data Objects )

$stmt = $pdo->prepare('SELECT * FROM employees WHERE name = :name');
$stmt->execute(array(':name' => $name));
foreach ($stmt as $row) {
    // do something with $row
}

2.使用mysqli

$stmt = $dbConnection->prepare('SELECT * FROM employees WHERE name = ?');
$stmt->bind_param('s', $name);
$stmt->execute();
$result = $stmt->get_result();
while ($row = $result->fetch_assoc()) {
    // do something with $row
}

PDO(PHP数据对象)

注意当使用PDO访问MySQL数据库真正的预备义语句并不是默认使用的!为了解决这个问题,你必须禁用仿真准备好的语句。使用PDO创建连接的例子如下:

$dbConnection = new PDO('mysql:dbname=dbtest;host=127.0.0.1;charset=utf8', 'user', 'pass');
$dbConnection->setAttribute(PDO::ATTR_EMULATE_PREPARES, false);
$dbConnection->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);

在上面例子中错误模式ERRMODE不是严格必须的,但是建议添加它。当运行出错产生致命错误时,这种方法脚本不会停止。并给开发人员捕捉任何错误的机会(当抛出PDOException异常时)。

setAttribute()那一行是强制性的,它告诉PDO禁用仿真预备义语句,使用真正的预备义语句。这可以确保语句和值在发送给MySQL数据库服务器前不被PHP解析(攻击者没有机会注入恶意的SQL).

当然你可以在构造函数选项中设置字符集参数,特别注意'老'的PHP版本(5.3.6)会在DSN中忽略掉字符集参数。

Explanation(解释)

在你传递的sql预备义语句 被数据库服务器解析和编译会发生什么?通过指定的字符(在上面例子中像a?或者像:name)告诉数据库引擎你想要过滤什么.然后调用execute执行结合好的预备义语句和你指定的参数值.

这里最重要的是,该参数值是和预编译的语句结合的,而不是和一个SQL字符串.SQL注入的工作原理是通过欺骗手段创建的SQL脚本包括恶意字符串发送到数据库.因此,通过发送实际的分开的sql参数,你会降低风险.使用准备好的语句时,你发送的任何参数,将只被视为字符串(虽然数据库引擎可能会做一些参数的优化,当然最终可能会为数字).在上面的例子中,如果变量$name包含'sarah';DELETE * FROM employees,结果只会是一个搜索的字符串"'sarah';DELETE * FROM employees",你不会得到一个空表。

使用准备好的语句的另一个好处是,如果你在同一会话中多次执行相同的语句,这将只被解析和编译一次,给你一些的速度增长。
哦,既然你问如何进行插入,这里是一个例子(使用PDO):

$preparedStatement = $db->prepare('INSERT INTO table (column) VALUES (:column)');
$preparedStatement->execute(array(':column' => $unsafeValue));

PHP 相关文章推荐
基于OpenCV的PHP图像人脸识别技术
Oct 11 PHP
PHP 学习路线与时间表
Feb 21 PHP
浅析PHP原理之变量(Variables inside PHP)
Aug 09 PHP
微信支付开发教程(一)微信支付URL配置
May 28 PHP
CodeIgniter启用缓存和清除缓存的方法
Jun 12 PHP
在Mac OS的PHP环境下安装配置MemCache的全过程解析
Feb 15 PHP
Zend Framework教程之Bootstrap类用法概述
Mar 14 PHP
ThinkPHP项目分组配置方法分析
Mar 23 PHP
PHP中静态变量的使用方法实例分析
Dec 01 PHP
PHP封装的page分页类定义与用法完整示例
Dec 24 PHP
laravel 获取当前url的别名方法
Oct 11 PHP
PHP连接SQL server数据库测试脚本运行实例
Aug 24 PHP
Apache下禁止php文件被直接访问的解决方案
Apr 25 #PHP
PHP笔记之:日期函数的使用介绍
Apr 24 #PHP
php笔记之:AOP的应用
Apr 24 #PHP
php class中self,parent,this的区别以及实例介绍
Apr 24 #PHP
PHP中::、->、self、$this几种操作符的区别介绍
Apr 24 #PHP
php判断终端是手机还是电脑访问网站的思路及代码
Apr 24 #PHP
用PHP提取中英文词语以及数字的首字母的方法介绍
Apr 23 #PHP
You might like
使用php4加速网络传输
2006/10/09 PHP
php下保存远程图片到本地的办法
2010/08/08 PHP
PHP分页函数代码(简单实用型)
2010/12/02 PHP
使用Java实现简单的server/client回显功能的方法介绍
2013/05/03 Javascript
Javascript实现带关闭按钮的网页漂浮广告代码
2014/01/12 Javascript
Egret引擎开发指南之视觉编程
2014/09/03 Javascript
jQuery实现Twitter的自动文字补齐特效
2014/11/28 Javascript
javascript经典特效分享 手风琴、轮播图、图片滑动
2016/09/14 Javascript
jQuery+CSS3实现仿花瓣网固定顶部位置带悬浮效果的导航菜单
2016/09/21 Javascript
实现隔行换色效果的两种方式【实用】
2016/11/27 Javascript
Bootstrap模态框插入视频的实现代码
2017/06/25 Javascript
JavaScript异步上传图片文件的实例代码
2017/07/04 Javascript
在vue中获取dom元素内容的方法
2017/07/10 Javascript
一个有意思的鼠标点击文字特效jquery代码
2017/09/23 jQuery
JS+H5 Canvas实现时钟效果
2018/07/20 Javascript
小程序如何在不同设备上自适应生成海报的实现方法
2019/08/20 Javascript
vue-resourc发起异步请求的方法
2020/02/11 Javascript
Vue项目配置跨域访问和代理proxy设置方式
2020/09/08 Javascript
Django框架下在视图中使用模版的方法
2015/07/16 Python
用python记录运行pid,并在需要时kill掉它们的实例
2017/01/16 Python
浅析python实现scrapy定时执行爬虫
2018/03/04 Python
python在openstreetmap地图上绘制路线图的实现
2019/07/11 Python
django执行数据库查询之后实现返回的结果集转json
2020/03/31 Python
基于tf.shape(tensor)和tensor.shape()的区别说明
2020/06/30 Python
python pygame 愤怒的小鸟游戏示例代码
2021/02/25 Python
Banana Republic英国官网:香蕉共和国,GAP集团旗下偏贵族风
2018/04/24 全球购物
英国最大的正宗复古足球衫制造商和零售商:TOFFS
2018/06/21 全球购物
Proenza Schouler官方网站:纽约女装和配饰品牌
2019/01/03 全球购物
Parfumdreams芬兰:购买香水和化妆品
2021/02/13 全球购物
以思科路由器为例你写下单臂路由的配置命令
2013/08/03 面试题
《云雀的心愿》教学反思
2014/02/25 职场文书
计算机专业职业规划
2014/02/28 职场文书
排查整治工作方案
2014/06/09 职场文书
浅谈Redis主从复制以及主从复制原理
2021/05/29 Redis
JS数组方法some、every和find的使用详情
2021/10/05 Javascript
浅谈Redis跟MySQL的双写问题解决方案
2022/02/24 Redis