探讨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下MYSQL limit的优化
Jan 10 PHP
php中修改浏览器的User-Agent来伪装你的浏览器和操作系统
Jul 29 PHP
LotusPhp笔记之:基于ObjectUtil组件的使用分析
May 06 PHP
分享一个超好用的php header下载函数
Jan 31 PHP
php中使用PHPExcel读写excel(xls)文件的方法
Sep 15 PHP
php截取html字符串及自动补全html标签的方法
Jan 15 PHP
Laravel给生产环境添加监听事件(SQL日志监听)
Jun 19 PHP
PHP编程文件处理类SplFileObject和SplFileInfo用法实例分析
Jul 22 PHP
PHP基于堆栈实现的高级计算器功能示例
Sep 15 PHP
php文件包含的几种方式总结
Sep 19 PHP
thinkphp 5框架实现登陆,登出及session登陆状态检测功能示例
Oct 10 PHP
PHP 对接美团大众点评团购券(门票)的开发步骤
Apr 03 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连接和操作MySQL数据库基础教程
2014/09/29 PHP
PHP实现链式操作的核心思想
2015/06/23 PHP
详解php的socket通信
2015/08/11 PHP
Laravel框架源码解析之模型Model原理与用法解析
2020/05/14 PHP
js 数组操作代码集锦
2009/04/28 Javascript
JS 文件大小判断的实现代码
2010/04/07 Javascript
jQuery插件开发全解析
2012/10/10 Javascript
JavaScript中跨域调用Flash的方法
2014/08/11 Javascript
把Node.js程序加入服务实现随机启动
2015/06/25 Javascript
jQuery实现微信长按识别二维码功能
2016/08/26 Javascript
基于 Vue 的树形选择组件的示例代码
2017/08/18 Javascript
VUE-cli3使用 svg-sprite-loader
2018/10/20 Javascript
微信小程序当前时间时段选择器插件使用方法详解
2018/12/28 Javascript
判断JavaScript中的两个变量是否相等的操作符
2019/12/21 Javascript
JavaScript中的this妙用实例分析
2020/05/09 Javascript
python连接数据库的方法
2017/10/19 Python
Python编程之Re模块下的函数介绍
2017/10/28 Python
对dataframe进行列相加,行相加的实例
2018/06/08 Python
Python实现网页截图(PyQT5)过程解析
2019/08/12 Python
将python文件打包exe独立运行程序方法详解
2020/02/12 Python
Python3 读取Word文件方式
2020/02/13 Python
python中time tzset()函数实例用法
2021/02/18 Python
加拿大著名时装品牌:SOIA & KYO
2016/08/23 全球购物
Weblogic的布署方式
2013/08/23 面试题
求职信格式范本
2013/11/15 职场文书
董事长秘书职责
2014/01/31 职场文书
《王二小》教学反思
2014/02/27 职场文书
经典英文广告词
2014/03/18 职场文书
紧急通知
2015/04/17 职场文书
离婚答辩状范文
2015/05/22 职场文书
教师考核鉴定意见
2015/06/05 职场文书
莫言诺贝尔获奖感言(全文)
2015/07/31 职场文书
Keras在mnist上的CNN实践,并且自定义loss函数曲线图操作
2021/05/25 Python
「天才王子的赤字国家重生术」妮妮姆·拉雷粘土人开订
2022/03/21 日漫
Linux系统下MySQL配置主从分离的步骤
2022/03/21 MySQL
前端框架ECharts dataset对数据可视化的高级管理
2022/12/24 Javascript