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 相关文章推荐
模仿OSO的论坛(一)
Oct 09 PHP
php使用ICQ网关发送手机短信
Oct 30 PHP
适用于抽奖程序、随机广告的PHP概率算法实例
Apr 09 PHP
ThinkPHP3.1新特性之动态设置自动完成及自动验证示例代码
Jun 23 PHP
PHP链接MySQL的常用扩展函数
Oct 23 PHP
PHP中变量引用与变量销毁机制分析
Nov 15 PHP
php字符串截取函数用法分析
Nov 25 PHP
PHP动态输出JavaScript代码实例
Feb 12 PHP
php lcg_value与mt_rand生成0~1随机小数的效果对比分析
Apr 05 PHP
php表单处理操作
Nov 16 PHP
Laravel5框架自定义错误页面配置操作示例
Apr 17 PHP
解决tp5在nginx下修改配置访问的问题
Oct 16 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
PHP 简单数组排序实现代码
2009/08/05 PHP
php文件服务实现虚拟挂载其他目录示例
2014/04/17 PHP
PHP滚动日志的代码实现
2015/06/10 PHP
Laravel日志用法详解
2016/10/09 PHP
浅谈Laravel POST,PUT,PATCH 路由的区别
2019/10/15 PHP
PHP7 其他修改
2021/03/09 PHP
Alliance vs Liquid BO3 第一场2.13
2021/03/10 DOTA
非主流的textarea自增长实现js代码
2011/12/20 Javascript
JS输入用户名自动显示邮箱后缀列表的方法
2015/01/27 Javascript
javascript实现框架高度随内容改变的方法
2015/07/23 Javascript
由浅入深剖析Angular表单验证
2016/07/14 Javascript
js捕捉键盘事件和按键键值的方法
2016/10/10 Javascript
jQuery上传多张图片带进度条样式(DEMO)
2017/03/02 Javascript
vue的Virtual Dom实现snabbdom解密
2017/05/03 Javascript
自定义事件解决重复请求BUG的问题
2017/07/11 Javascript
Element Table的row-class-name无效与动态高亮显示选中行背景色
2018/11/30 Javascript
Vue+elementui 实现复杂表头和动态增加列的二维表格功能
2019/09/23 Javascript
H5 js点击按钮复制文本到粘贴板
2020/11/19 Javascript
[01:03:56]Mineski vs TNC 2018国际邀请赛淘汰赛BO1 8.21
2018/08/22 DOTA
python自定义解析简单xml格式文件的方法
2015/05/11 Python
Python实现二分查找算法实例
2015/05/26 Python
python根据京东商品url获取产品价格
2015/08/09 Python
Python网络编程详解
2017/10/31 Python
pandas中的DataFrame按指定顺序输出所有列的方法
2018/04/10 Python
python下解压缩zip文件并删除文件的实例
2018/04/24 Python
Python单元测试unittest的具体使用示例
2018/12/17 Python
Python绘制热力图示例
2019/09/27 Python
Python爬虫过程解析之多线程获取小米应用商店数据
2020/11/14 Python
简单几步用纯CSS3实现3D翻转效果
2019/01/17 HTML / CSS
HTML5中div、article、section的区别及使用介绍
2013/08/14 HTML / CSS
工作会议欢迎词
2014/01/16 职场文书
高中军训感想300字
2014/03/04 职场文书
降价通知函
2015/04/23 职场文书
《倍数和因数》教学反思
2016/02/23 职场文书
pytorch 使用半精度模型部署的操作
2021/05/24 Python
详解NumPy中的线性关系与数据修剪压缩
2022/05/25 Python