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冒泡排序算法的深入理解
Jun 09 PHP
smarty基础之拼接字符串的详解
Jun 18 PHP
PHP生成迅雷、快车、旋风等软件的下载链接代码实例
May 12 PHP
php字符比较函数similar_text、strnatcmp与strcasecmp用法分析
Nov 18 PHP
ubuntu下配置nginx+php+mysql详解
Sep 10 PHP
PHP远程调试之XDEBUG
Dec 29 PHP
PHP 自动加载的简单实现(推荐)
Aug 12 PHP
Laravel框架路由和控制器的绑定操作方法
Jun 12 PHP
PHP实现文字写入图片功能
Feb 18 PHP
PHP进阶学习之依赖注入与Ioc容器详解
Jun 19 PHP
php和nginx交互实例讲解
Sep 24 PHP
Laravel5.1 框架Request请求操作常见用法实例分析
Jan 04 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&&mysql)六
2006/10/09 PHP
浅析php中常量,变量的作用域和生存周期
2013/08/10 PHP
PHP实现的简单AES加密解密算法实例
2017/05/29 PHP
Laravel中GraphQL接口请求频率实战记录
2020/09/01 PHP
windows系统php环境安装swoole具体步骤
2021/03/04 PHP
JQuery扩展插件Validate 3通过参数设置错误信息
2011/09/05 Javascript
jquery $.each 和for怎么跳出循环终止本次循环
2013/09/27 Javascript
加载列表时jquery获取ul中第一个li的属性
2014/11/02 Javascript
JavaScript中解析JSON数据的三种方法
2015/07/03 Javascript
使用jquery插件qrcode生成二维码
2015/10/22 Javascript
浅谈jquery点击label触发2次的问题
2016/06/12 Javascript
ReactNative-JS 调用原生方法实例代码
2016/10/08 Javascript
javascript中href和replace的比较(详解)
2016/11/25 Javascript
Bootstrap如何激活导航状态
2017/03/22 Javascript
JS实现的四级密码强度检测功能示例
2017/05/11 Javascript
详解用Node.js写一个简单的命令行工具
2018/03/01 Javascript
深入理解 Koa 框架中间件原理
2018/10/18 Javascript
Node.js使用MongoDB的ObjectId作为查询条件的方法
2019/09/10 Javascript
jquery实现简单拖拽效果
2020/07/20 jQuery
[01:14]2019完美世界城市挑战赛(秋季赛)全国总决赛精彩花絮
2020/01/08 DOTA
Python_LDA实现方法详解
2017/10/25 Python
使用Python爬取最好大学网大学排名
2018/02/24 Python
Win10下python3.5和python2.7环境变量配置教程
2018/09/18 Python
Python+PyQt5实现美剧爬虫可视工具的方法
2019/04/25 Python
python对常见数据类型的遍历解析
2019/08/27 Python
Pycharm最新激活码2019(推荐)
2019/12/31 Python
Python Http请求json解析库用法解析
2020/11/28 Python
python3判断IP地址的方法
2021/03/04 Python
html5实现完美兼容各大浏览器的播放器
2014/12/26 HTML / CSS
如何用H5实现一个触屏版的轮播器的实例
2017/01/09 HTML / CSS
俄罗斯香水和化妆品网上商店:NOTINO.ru
2019/12/17 全球购物
打架检讨书100字
2014/01/19 职场文书
上课玩手机检讨书
2014/02/08 职场文书
英文道歉信
2015/01/20 职场文书
nginx限制并发连接请求数的方法
2021/04/01 Servers
MongoDB数据库部署环境准备及使用介绍
2022/03/21 MongoDB