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 相关文章推荐
PHP中for循环语句的几种变型
Mar 16 PHP
PHP开发不能违背的安全规则 过滤用户输入
May 01 PHP
ThinkPHP使用心得分享-上传类UploadFile的使用
May 15 PHP
PHP语法自动检查的Vim插件
Aug 11 PHP
ThinkPHP通过AJAX返回JSON的两种实现方法
Dec 18 PHP
Thinkphp中的curd应用实用要点
Jan 04 PHP
php实现的美国50个州选择列表实例
Apr 20 PHP
PHP 读取大文件并显示的简单实例(推荐)
Aug 12 PHP
PHP使用strrev翻转中文乱码问题的解决方法
Jan 13 PHP
老生常谈文本文件和二进制文件的区别
Feb 27 PHP
PHP PDOStatement::setAttribute讲解
Feb 01 PHP
基于thinkphp5框架实现微信小程序支付 退款 订单查询 退款查询操作
Aug 17 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
CentOS 6.2使用yum安装LAMP以及phpMyadmin详解
2013/06/17 PHP
php实现mysql数据库操作类分享
2014/02/14 PHP
PHP也能干大事之PHP中的编码解码详解
2015/04/20 PHP
php分页查询的简单实现代码
2017/03/14 PHP
PHP删除字符串中非字母数字字符方法总结
2019/01/20 PHP
PHP fopen函数用法实例讲解
2019/02/15 PHP
PHP实现发送微博消息功能完整示例
2019/12/04 PHP
[原创]网络复制内容时常用的正则+editplus
2006/11/30 Javascript
Html中JS脚本执行顺序简单举例说明
2010/06/19 Javascript
20个非常棒的 jQuery 幻灯片插件和教程分享
2011/08/23 Javascript
javascript中substring()、substr()、slice()的区别
2015/08/30 Javascript
js判断文本框输入的内容是否为数字
2015/12/23 Javascript
AngularJS基础 ng-options 指令详解
2016/08/02 Javascript
基于MVC+EasyUI的web开发框架之使用云打印控件C-Lodop打印页面或套打报关运单信息
2016/08/29 Javascript
js实现简单的手风琴效果
2017/02/27 Javascript
vue.js利用defineProperty实现数据的双向绑定
2017/04/28 Javascript
php 修改密码实现代码
2017/05/24 Javascript
SpringMVC简单整合Angular2的示例
2017/07/31 Javascript
js实现扫雷小程序的示例代码
2017/09/27 Javascript
深入理解Angularjs 脏值检测
2018/10/12 Javascript
微信小程序scroll-view锚点链接滚动跳转功能
2019/12/12 Javascript
vue实现简单跑马灯效果
2020/05/25 Javascript
react实现复选框全选和反选组件效果
2020/08/25 Javascript
[46:27]DOTA2上海特级锦标赛主赛事日 - 1 胜者组第一轮#2LGD VS MVP.Phx第一局
2016/03/02 DOTA
python opencv 读取本地视频文件 修改ffmpeg的方法
2019/01/26 Python
python shutil文件操作工具使用实例分析
2019/12/25 Python
python3.8下载及安装步骤详解
2020/01/15 Python
Python3打包exe代码2种方法实例解析
2020/02/17 Python
One.com挪威:北欧成长最快的网络托管公司
2016/11/19 全球购物
苹果台湾官网:Apple台湾
2019/01/05 全球购物
校长岗位职责
2013/11/26 职场文书
七夕活动策划方案
2014/08/16 职场文书
项目建议书
2015/02/04 职场文书
2015年中学团委工作总结
2015/07/22 职场文书
先进教师个人主要事迹材料
2015/11/03 职场文书
详解MySQL的半同步
2021/04/22 MySQL