探讨php中防止SQL注入最好的方法是什么


Posted in PHP onJune 10, 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;'

我们应该怎么防止这种情况呢?请看下面
使用预备义语句和参数化查询。对于带有任何参数的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 相关文章推荐
php通过COM类调用组件的实现代码
Jan 11 PHP
php实现统计邮件大小的方法
Aug 06 PHP
PHP图像处理类库MagickWand用法实例分析
May 21 PHP
typecho插件编写教程(五):核心代码
May 28 PHP
php自定义分页类完整实例
Dec 25 PHP
PHP实现微信网页授权开发教程
Jan 19 PHP
Laravel 5.3 学习笔记之 配置
Aug 28 PHP
PHP实现的折半查询算法示例
Oct 09 PHP
PHP递归的三种常用方式
Feb 28 PHP
thinkphp5.1框架中容器(Container)和门面(Facade)的实现方法分析
Aug 05 PHP
Laravel 不同生产环境服务器的判断实践
Oct 15 PHP
php文件上传原理与实现方法详解
Dec 20 PHP
php防注入,表单提交值转义的实现详解
Jun 10 #PHP
PHP获取当前页面完整URL的实现代码
Jun 10 #PHP
如何判断php数组的维度
Jun 10 #PHP
joomla jce editor 解决上传中文名文件失败问题
Jun 09 #PHP
php缩放图片(根据宽高的等比例缩放)实例介绍
Jun 09 #PHP
探讨php define()函数及defined()函数使用详解
Jun 09 #PHP
如何用php获取程序执行的时间
Jun 09 #PHP
You might like
php session 写入数据库
2016/02/13 PHP
Yii使用migrate命令执行sql语句的方法
2016/03/15 PHP
PHPstorm快捷键(分享)
2017/07/17 PHP
PHP xpath提取网页数据内容代码解析
2020/07/16 PHP
在Javascript中定义对象类别
2006/12/22 Javascript
在页面上点击任一链接时触发一个事件的代码
2007/04/07 Javascript
javascript使用定时函数实现跳转到某个页面
2013/12/25 Javascript
JavaScript实现标题栏文字轮播效果代码
2015/10/24 Javascript
Bootstrap入门书籍之(三)栅格系统
2016/02/17 Javascript
JS封装通过className获取元素的函数示例
2016/12/20 Javascript
AngularJS的ng-repeat指令与scope继承关系实例详解
2017/01/21 Javascript
js Date()日期函数浏览器兼容问题解决方法
2017/09/12 Javascript
基于对象合并功能的实现示例
2017/10/10 Javascript
CheckBox多选取值及判断CheckBox选中是否为空的实例
2017/10/31 Javascript
vue做移动端适配最佳解决方案(亲测有效)
2018/09/04 Javascript
js编写简易的计算器
2020/07/29 Javascript
从表单校验看JavaScript策略模式的使用详解
2020/10/17 Javascript
深入理解Python中的内置常量
2017/05/20 Python
PyTorch基本数据类型(一)
2019/05/22 Python
python是否适合网页编程详解
2019/10/04 Python
Python集合操作方法详解
2020/02/09 Python
python 自定义异常和主动抛出异常(raise)的操作
2020/12/11 Python
纯CSS3实现手风琴风格菜单具体步骤
2013/05/06 HTML / CSS
用HTML5制作烟火效果的教程
2015/05/12 HTML / CSS
娱乐地球:Entertainment Earth
2020/01/08 全球购物
澳大利亚头发和美容产品购物网站:OZ Hair & Beauty
2020/03/27 全球购物
Jar包的作用是什么
2014/03/30 面试题
廉政教育心得体会
2014/01/01 职场文书
网站美工岗位职责
2014/04/02 职场文书
大学生学习面向未来的赶考思想汇报
2014/09/12 职场文书
2014年政协委员工作总结
2014/12/01 职场文书
2014年大学教师工作总结
2014/12/02 职场文书
党员志愿者服务倡议书
2015/04/29 职场文书
行政后勤人员工作计划应该怎么写?
2019/08/16 职场文书
浅谈Redis存储数据类型及存取值方法
2021/05/08 Redis
nginx之内存池的实现
2022/06/28 Servers