探讨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防注入安全代码
Apr 09 PHP
phpMyadmin 用户权限中英对照
Apr 02 PHP
C#静态方法与非静态方法实例分析
Sep 22 PHP
DOM基础及php读取xml内容操作的方法
Jan 23 PHP
php遍历删除整个目录及文件的方法
Mar 13 PHP
YII2.0之Activeform表单组件用法实例
Jan 09 PHP
smarty中改进truncate使其支持中文的方法
May 30 PHP
php页面跳转session cookie丢失导致不能登录等问题的解决方法
Dec 12 PHP
PHP基于PDO实现的SQLite操作类【包含增删改查及事务等操作】
Jun 21 PHP
php处理多图上传压缩代码功能
Jun 13 PHP
PHP Ajax跨域问题解决方案代码实例
Aug 01 PHP
php框架知识点的整理和补充
Mar 01 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
比特率,大家看看这个就不用收音机音质去比MP3音质了
2021/03/01 无线电
php 禁止页面缓存输出
2009/01/07 PHP
PHP和MySql中32位和64位的整形范围是多少
2016/02/18 PHP
点击下载链接 弹出页面实现代码
2009/10/01 Javascript
js实现的日期操作类DateTime函数代码
2010/03/16 Javascript
jQuery 核心函数以及jQuery对象
2010/03/23 Javascript
IE和Firefox的Javascript兼容性总结[推荐收藏]
2011/10/19 Javascript
javascript中onclick(this)用法介绍
2013/04/19 Javascript
js 一个关于图片onload加载的事
2013/11/10 Javascript
Javascript中3种实现继承的方法和代码实例
2014/08/12 Javascript
使用jQuery在对象中缓存选择器的简单方法
2015/06/30 Javascript
js实现简单折叠、展开菜单的方法
2015/08/28 Javascript
JavaScript几种数组去掉重复值的方法推荐
2016/04/12 Javascript
JQuery在循环中绑定事件的问题详解
2016/06/02 Javascript
JS树形菜单组件Bootstrap TreeView使用方法详解
2016/12/21 Javascript
基于jQuery实现的打字机效果
2017/01/16 Javascript
Angular2学习教程之ng中变更检测问题详解
2017/05/28 Javascript
Angular4 中常用的指令入门总结
2017/06/12 Javascript
vue中SPA单页面应用程序详解
2017/11/07 Javascript
vue组件创建的三种方式小结
2020/02/03 Javascript
javascript实现智能手环时间显示
2020/09/18 Javascript
[50:20]DOTA2上海特级锦标赛主赛事日 - 5 总决赛Liquid VS Secret第四局
2016/03/06 DOTA
Python实现新浪博客备份的方法
2016/04/27 Python
django启动uwsgi报错的解决方法
2018/04/08 Python
解析python 中/ 和 % 和 //(地板除)
2020/06/28 Python
python中strip(),lstrip(),rstrip()函数的使用讲解
2020/11/17 Python
基于HTML5 Canvas:字符串,路径,背景,图片的详解
2013/05/09 HTML / CSS
Shell如何接收变量输入
2012/09/24 面试题
校本教研工作方案
2014/01/14 职场文书
聊城大学毕业生自荐书
2014/02/01 职场文书
学校春季防火方案
2014/06/08 职场文书
先进班组材料范文
2014/12/25 职场文书
房产公证书格式
2015/01/26 职场文书
2015迎新晚会活动总结
2015/07/16 职场文书
职工宿舍管理制度
2015/08/05 职场文书
关于ObjectUtils.isEmpty() 和 null 的区别
2022/02/28 Java/Android