background image

$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

));