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 中的类
Oct 09 PHP
解析php做推送服务端实现ios消息推送
Jul 01 PHP
ThinkPHP中使用ajax接收json数据的方法
Dec 18 PHP
用php来限制每个ip每天浏览页面数量的实现思路
Feb 24 PHP
PHP程序中使用adodb连接不同数据库的代码实例
Dec 19 PHP
PHP调试的强悍利器之PHPDBG
Feb 22 PHP
PHP Static延迟静态绑定用法分析
Mar 16 PHP
PHP判断FORM表单或URL参数来的数据是否为整数的方法
Mar 25 PHP
PHP receiveMail实现收邮件功能
Apr 25 PHP
PHP简单实现图片格式转换(jpg转png,gif转png等)
Oct 30 PHP
PHP执行普通shell命令流程解析
Aug 24 PHP
phpstorm激活码2020附使用详细教程
Sep 25 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
用文本文件制作留言板提示(上)
2006/10/09 PHP
php学习笔记之 函数声明(二)
2011/06/09 PHP
php设计模式 Strategy(策略模式)
2011/06/26 PHP
简单的php+mysql聊天室实现方法(附源码)
2016/01/05 PHP
JavaScript 动态创建VML的方法
2009/10/14 Javascript
jquery选择器、属性设置用法经验总结
2013/09/08 Javascript
非常酷炫的Bootstrap图片轮播动画
2016/05/27 Javascript
深入理解Javascript中的自执行匿名函数
2016/06/03 Javascript
JS数组去掉重复数据只保留一条的实现代码
2016/08/11 Javascript
js获取form表单所有数据的简单方法
2016/08/18 Javascript
EditPlus 正则表达式 实战(3)
2016/12/15 Javascript
node.js基于mongodb的搜索分页示例
2017/01/22 Javascript
js css3实现图片拖拽效果
2017/03/04 Javascript
Bootstrap Multiselect 常用组件实现代码
2017/07/09 Javascript
jQuery列表检索功能实现代码
2017/07/17 jQuery
jQuery实现的五星点评功能【案例】
2019/02/18 jQuery
详解Vscode中使用Eslint终极配置大全
2019/11/08 Javascript
python用ConfigObj读写配置文件的实现代码
2013/03/04 Python
python函数装饰器用法实例详解
2015/06/04 Python
django数据库migrate失败的解决方法解析
2018/02/08 Python
Python连接Redis的基本配置方法
2018/09/13 Python
PyQt5中QTableWidget如何弹出菜单的示例代码
2020/02/23 Python
Python+OpenCV检测灯光亮点的实现方法
2020/11/02 Python
解决pytorch下出现multi-target not supported at的一种可能原因
2021/02/06 Python
Pytorch模型迁移和迁移学习,导入部分模型参数的操作
2021/03/03 Python
Brookstone美国官网:独特新奇产品
2017/03/04 全球购物
Tessabit美国:集世界奢侈品和设计师品牌的意大利精品买手店
2020/06/29 全球购物
电子商务专业实习生自我鉴定
2013/09/24 职场文书
终端业务员岗位职责
2013/11/27 职场文书
24岁生日感言
2014/01/13 职场文书
优秀食品类广告词
2014/03/19 职场文书
模具专业自荐信
2014/05/29 职场文书
2014年职称评定工作总结
2014/11/26 职场文书
婚礼父母答谢词
2015/01/04 职场文书
MySQL InnoDB ReplicaSet(副本集)简单介绍
2021/04/24 MySQL
HTML+css盒子模型案例(圆,半圆等)“border-radius” 简单易上手
2021/05/10 HTML / CSS