mysql_*
为什么不应该使用函数的技术原因是什么?(例如mysql_query()
,mysql_connect()
或mysql_real_escape_string()
)?
即使它们在我的网站上工作,我为什么还要使用其他东西?
如果他们在我的网站上不起作用,为什么我会收到类似的错误
警告:mysql_connect():没有这样的文件或目录
MySQL 扩展:
由于它已被弃用,因此使用它会使您的代码不那么面向未来。
缺乏对准备好的语句的支持尤其重要,因为它们提供了一种更清晰、更不容易出错的方法来转义和引用外部数据,而不是使用单独的函数调用手动转义它。
请参阅SQL 扩展的比较。
PHP 提供了三种不同的 API 来连接 MySQL。这些是mysql
(从 PHP 7 中删除)mysqli
、 和PDO
扩展。
这些mysql_*
功能曾经非常流行,但不再鼓励使用它们。文档团队正在讨论数据库安全情况,教育用户远离常用的 ext/mysql 扩展是其中的一部分(查看php.internals: deprecating ext/mysql)。
而后来的 PHP 开发团队已经决定E_DEPRECATED
在用户连接 MySQL 时产生错误,无论mysql_connect()
是mysql_pconnect()
通过ext/mysql
.
ext/mysql
自 PHP 5.5起正式弃用,自 PHP 7起已被删除。
看到红盒子了吗?
当您进入任何mysql_*
功能手册页时,您会看到一个红色框,说明不应再使用它。
远离ext/mysql
不仅关乎安全性,还关乎能够访问 MySQL 数据库的所有功能。
ext/mysql
是为MySQL 3.23构建的,从那时起只添加了很少的内容,同时主要保持与这个旧版本的兼容性,这使得代码更难维护。包括不支持的缺失功能ext/mysql
:(来自 PHP 手册)。
不使用mysql_*
功能的原因:
缺乏对准备好的语句的支持尤其重要,因为它们提供了一种更清晰、更不容易出错的方法来转义和引用外部数据,而不是使用单独的函数调用手动转义它。
请参阅SQL 扩展的比较。
抑制弃用警告
在将代码转换为MySQLi
/PDO
时,E_DEPRECATED
可以通过error_reporting
在php.ini中设置 excludeE_DEPRECATED:
error_reporting = E_ALL ^ E_DEPRECATED
请注意,这也会隐藏其他弃用警告,但是,这些警告可能适用于 MySQL 以外的东西。(来自 PHP 手册)
文章PDO 与 MySQLi:您应该使用哪个?Dejan Marjanovic将帮助您选择。
一个更好的方法是PDO
,我现在正在写一个简单的PDO
教程。
A. “<strong>PDO – PHP 数据对象 – 是一个数据库访问层,提供了访问多个数据库的统一方法。”</p>
使用mysql_*
函数或者我们可以说它是旧的方式(在 PHP 5.5 及更高版本中已弃用)
$link = mysql_connect('localhost', 'user', 'pass');
mysql_select_db('testdb', $link);
mysql_set_charset('UTF-8', $link);
With PDO
:您需要做的就是创建一个新PDO
对象。构造函数接受用于指定数据库源PDO
的参数 构造函数主要接受四个参数,它们是DSN
(数据源名称)和可选username
的,password
。
在这里,我认为您对所有内容都很熟悉,除了DSN
; 这是新的PDO
。ADSN
基本上是一串选项,用于告诉PDO
要使用哪个驱动程序以及连接详细信息。如需进一步参考,请查看PDO MySQL DSN。
$db = new PDO('mysql:host=localhost;dbname=testdb;charset=utf8', 'username', 'password');
注意:你也可以使用charset=UTF-8
,但有时会导致错误,所以最好使用utf8
。
如果有任何连接错误,它会抛出一个PDOException
可以被捕获以Exception
进一步处理的对象。
好读:连接和连接管理¶
您还可以将多个驱动程序选项作为数组传递给第四个参数。我建议传递PDO
进入异常模式的参数。因为某些PDO
驱动程序不支持本机prepared statements,所以PDO
执行prepare 的模拟。它还允许您手动启用此仿真。要使用本机服务器端准备好的语句,您应该显式设置它false
。
另一种是关闭MySQL
驱动中默认开启的prepare emulation,但是prepare emulation应该关闭才能PDO
安全使用。
稍后我将解释为什么应该关闭准备模拟。要查找原因,请查看此帖子。
仅当您使用MySQL
我不推荐的旧版本时才可用。
下面是一个如何做到这一点的例子:
$db = new PDO('mysql:host=localhost;dbname=testdb;charset=UTF-8',
'username',
'password',
array(PDO::ATTR_EMULATE_PREPARES => false,
PDO::ATTR_ERRMODE => PDO::ERRMODE_EXCEPTION));
我们可以在 PDO 构建后设置属性吗?
是的,我们也可以在 PDO 构建后设置一些属性,setAttribute
方法如下:
$db = new PDO('mysql:host=localhost;dbname=testdb;charset=UTF-8',
'username',
'password');
$db->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
$db->setAttribute(PDO::ATTR_EMULATE_PREPARES, false);
错误处理PDO
比mysql_*
.
使用时的常见做法mysql_*
是:
//Connected to MySQL
$result = mysql_query("SELECT * FROM table", $link) or die(mysql_error($link));
OR die()
不是处理错误的好方法,因为我们无法处理die
. 它只会突然结束脚本,然后将错误回显到您通常不想向最终用户显示的屏幕上,并让该死的黑客发现您的架构。或者,mysql_*
函数的返回值通常可以与mysql_error()一起使用来处理错误。
PDO
提供了一个更好的解决方案:异常。我们所做的任何事情都PDO
应该包装在一个try
-catch
块中。我们可以PDO
通过设置错误模式属性来强制进入三种错误模式之一。以下是三种错误处理模式。
PDO::ERRMODE_SILENT
. mysql_*
它只是设置错误代码,其行为与您必须检查每个结果然后查看$db->errorInfo();
以获取错误详细信息的位置几乎相同。PDO::ERRMODE_WARNING
提高E_WARNING
。(运行时警告(非致命错误)。脚本的执行不会停止。)PDO::ERRMODE_EXCEPTION
: 抛出异常。它表示 PDO 引发的错误。PDOException
你不应该从你自己的代码中抛出一个。有关 PHP 中的异常的更多信息,请参阅异常。or die(mysql_error());
当它未被捕获时,它的行为非常类似于。但与 不同的是,如果您选择这样做or die()
,则可以优雅地捕获和处理。PDOException
好读:
像:
$stmt->setAttribute( PDO::ATTR_ERRMODE, PDO::ERRMODE_SILENT );
$stmt->setAttribute( PDO::ATTR_ERRMODE, PDO::ERRMODE_WARNING );
$stmt->setAttribute( PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION );
您可以将其包装在try
-catch
中,如下所示:
try {
//Connect as appropriate as above
$db->query('hi'); //Invalid query!
}
catch (PDOException $ex) {
echo "An Error occured!"; //User friendly message/message you want to show to user
some_logging_function($ex->getMessage());
}
您现在不必处理try
- catch
。您可以在任何适当的时候捕捉到它,但我强烈建议您使用try
- catch
。PDO
此外,在调用这些东西的函数之外捕获它可能更有意义:
function data_fun($db) {
$stmt = $db->query("SELECT * FROM table");
return $stmt->fetchAll(PDO::FETCH_ASSOC);
}
//Then later
try {
data_fun($db);
}
catch(PDOException $ex) {
//Here you can handle error and show message/perform action you want.
}
此外,您可以处理 byor die()
或者我们可以说 like mysql_*
,但它会非常多样化。您可以通过display_errors off
阅读错误日志来隐藏生产中的危险错误消息。
现在,在阅读了以上所有内容之后,您可能在想:当我只想开始学习简单SELECT
的 , INSERT
, UPDATE
, 或DELETE
语句时,这到底是什么鬼?别着急,我们来:
所以你正在做的mysql_*
是:
<?php
$result = mysql_query('SELECT * from table') or die(mysql_error());
$num_rows = mysql_num_rows($result);
while($row = mysql_fetch_assoc($result)) {
echo $row['field1'];
}
现在PDO
,您可以这样做:
<?php
$stmt = $db->query('SELECT * FROM table');
while($row = $stmt->fetch(PDO::FETCH_ASSOC)) {
echo $row['field1'];
}
或者
<?php
$stmt = $db->query('SELECT * FROM table');
$results = $stmt->fetchAll(PDO::FETCH_ASSOC);
//Use $results
注意:如果您使用如下 ( query()
) 的方法,此方法将返回一个PDOStatement
对象。因此,如果您想获取结果,请像上面一样使用它。
<?php
foreach($db->query('SELECT * FROM table') as $row) {
echo $row['field1'];
}
在 PDO Data 中,它是通过->fetch()
语句句柄的 , 方法获得的。在调用 fetch 之前,最好的方法是告诉 PDO 您希望如何获取数据。在下面的部分中,我将对此进行解释。
注意上面和代码PDO::FETCH_ASSOC
中的使用。这告诉以字段名称作为键的关联数组返回行。还有许多其他的获取模式,我将一一解释。fetch()
fetchAll()
PDO
首先,我解释一下如何选择获取模式:
$stmt->fetch(PDO::FETCH_ASSOC)
在上面,我一直在使用fetch()
. 您还可以使用:
PDOStatement::fetchAll()
- 返回一个包含所有结果集行的数组PDOStatement::fetchColumn()
- 从结果集的下一行返回单列PDOStatement::fetchObject()
- 获取下一行并将其作为对象返回。PDOStatement::setFetchMode()
- 设置此语句的默认获取模式现在我来获取模式:
PDO::FETCH_ASSOC
: 返回一个按列名索引的数组,与结果集中返回的一样PDO::FETCH_BOTH
(默认):返回一个由列名和 0 索引列号索引的数组,如结果集中返回的那样还有更多选择!PDOStatement
在Fetch 文档中阅读所有内容。.
获取行数:
您可以获取 a和 do ,而不是使用mysql_num_rows
来获取返回的行数,例如:PDOStatement
rowCount()
<?php
$stmt = $db->query('SELECT * FROM table');
$row_count = $stmt->rowCount();
echo $row_count.' rows selected';
获取最后插入的 ID
<?php
$result = $db->exec("INSERT INTO table(firstname, lastname) VAULES('John', 'Doe')");
$insertId = $db->lastInsertId();
我们在mysql_*
函数中所做的是:
<?php
$results = mysql_query("UPDATE table SET field='value'") or die(mysql_error());
echo mysql_affected_rows($result);
在 pdo 中,同样的事情可以通过以下方式完成:
<?php
$affected_rows = $db->exec("UPDATE table SET field='value'");
echo $affected_rows;
在上面的查询中PDO::exec
执行一条 SQL 语句并返回受影响的行数。
插入和删除将在后面介绍。
上述方法仅在您未在查询中使用变量时有用。但是当您需要在查询中使用变量时,永远不要像上面那样尝试 准备好的语句或参数化的语句。
问:什么是准备好的陈述,我为什么需要它们?
A.预编译语句是预编译的 SQL 语句,只需将数据发送到服务器即可多次执行。
使用prepared statement的典型工作流程如下(引自维基百科三三点):
准备:报表模板由应用程序创建并发送到数据库管理系统 (DBMS)。某些值未指定,称为参数、占位符或绑定变量(?
如下所示):
INSERT INTO PRODUCT (name, price) VALUES (?, ?)
DBMS 对语句模板进行解析、编译和查询优化,并存储结果而不执行它。
1.00
在此示例中,它可能为第一个参数和第二个参数提供“面包” 。您可以通过在 SQL 中包含占位符来使用准备好的语句。基本上有三个没有占位符(不要尝试使用上面的变量),一个带有未命名的占位符,一个带有命名的占位符。
问:那么现在,什么是命名占位符以及如何使用它们?
A.命名占位符。使用以冒号开头的描述性名称,而不是问号。我们不关心名称占位符中的位置/价值顺序:
$stmt->bindParam(':bla', $bla);
bindParam(parameter,variable,data_type,length,driver_options)
您也可以使用执行数组进行绑定:
<?php
$stmt = $db->prepare("SELECT * FROM table WHERE id=:id AND name=:name");
$stmt->execute(array(':name' => $name, ':id' => $id));
$rows = $stmt->fetchAll(PDO::FETCH_ASSOC);
对于朋友来说,另一个不错的功能OOP
是命名占位符能够将对象直接插入到您的数据库中,假设属性与命名字段匹配。例如:
class person {
public $name;
public $add;
function __construct($a,$b) {
$this->name = $a;
$this->add = $b;
}
}
$demo = new person('john','29 bla district');
$stmt = $db->prepare("INSERT INTO table (name, add) value (:name, :add)");
$stmt->execute((array)$demo);
问:那么现在,什么是未命名的占位符,我该如何使用它们?
A.举个例子:
<?php
$stmt = $db->prepare("INSERT INTO folks (name, add) values (?, ?)");
$stmt->bindValue(1, $name, PDO::PARAM_STR);
$stmt->bindValue(2, $add, PDO::PARAM_STR);
$stmt->execute();
和
$stmt = $db->prepare("INSERT INTO folks (name, add) values (?, ?)");
$stmt->execute(array('john', '29 bla district'));
在上面,您可以看到那些?
而不是名称占位符中的名称。现在在第一个示例中,我们将变量分配给各种占位符 ( $stmt->bindValue(1, $name, PDO::PARAM_STR);
)。然后,我们为这些占位符赋值并执行语句。在第二个示例中,第一个数组元素转到第一个?
,第二个转到第二个?
。
注意:在未命名的占位符中,我们必须注意传递给PDOStatement::execute()
方法的数组中元素的正确顺序。
SELECT
, INSERT
, UPDATE
,DELETE
准备好的查询SELECT
:
$stmt = $db->prepare("SELECT * FROM table WHERE id=:id AND name=:name");
$stmt->execute(array(':name' => $name, ':id' => $id));
$rows = $stmt->fetchAll(PDO::FETCH_ASSOC);
INSERT
:
$stmt = $db->prepare("INSERT INTO table(field1,field2) VALUES(:field1,:field2)");
$stmt->execute(array(':field1' => $field1, ':field2' => $field2));
$affected_rows = $stmt->rowCount();
DELETE
:
$stmt = $db->prepare("DELETE FROM table WHERE id=:id");
$stmt->bindValue(':id', $id, PDO::PARAM_STR);
$stmt->execute();
$affected_rows = $stmt->rowCount();
UPDATE
:
$stmt = $db->prepare("UPDATE table SET name=? WHERE id=?");
$stmt->execute(array($name, $id));
$affected_rows = $stmt->rowCount();
然而PDO
和/或MySQLi
并不完全安全。检查答案PDO 准备好的语句是否足以防止 SQL 注入?通过ircmaxell。另外,我从他的回答中引用了一部分:
$pdo->setAttribute(PDO::ATTR_EMULATE_PREPARES, false);
$pdo->query('SET NAMES GBK');
$stmt = $pdo->prepare("SELECT * FROM test WHERE name = ? LIMIT 1");
$stmt->execute(array(chr(0xbf) . chr(0x27) . " OR 1=1 /*"));
首先,让我们从我们给大家的标准评论开始:
请不要
mysql_*
在新代码中使用函数。它们不再被维护并被正式弃用。看到红框了吗?改为了解准备好的语句,并使用PDO或MySQLi -本文将帮助您决定使用哪个。如果您选择 PDO,这里有一个很好的教程。
让我们逐句解释一下:
它们不再维护,并已正式弃用
这意味着 PHP 社区正在逐渐放弃对这些非常古老的功能的支持。它们很可能不会出现在 PHP 的未来(最近)版本中!继续使用这些功能可能会在(不是那么)遥远的将来破坏您的代码。
新的!-自 PHP 5.5 起, ext/mysql 现已正式弃用!
相反,您应该了解准备好的语句
mysql_*
扩展不支持准备好的语句,这是(除其他外)对SQL 注入非常有效的对策。它修复了 MySQL 相关应用程序中的一个非常严重的漏洞,该漏洞允许攻击者访问您的脚本并对您的数据库执行任何可能的查询。
有关更多信息,请参阅如何防止 PHP 中的 SQL 注入?
看到红盒子了吗?
当您转到任何mysql
功能手册页时,您会看到一个红色框,说明不应再使用它。
使用 PDO 或 MySQLi
有更好、更健壮和构建良好的替代方案,PDO - PHP 数据库对象,它为数据库交互提供了完整的 OOP 方法,以及MySQLi,它是 MySQL 的特定改进。
分析原因和综合原因已经提到。对于新手来说,停止使用过时的 mysql_ 函数有更大的动力。
现代数据库 API更易于使用。
它主要是可以简化代码的绑定参数。借助优秀的教程(如上所示),向PDO的过渡并不过分艰巨。
然而,一次重写更大的代码库需要时间。这种中间选择的存在理由:
使用< pdo_mysql.php >您可以毫不费力地从旧的 mysql_ 函数切换。它添加了pdo_
替换其mysql_
对应物的函数包装器。
只需在必须与数据库交互的每个调用脚本中。
include_once(
"pdo_mysql.php"
);
删除各处的函数前缀并将其替换为.mysql_
pdo_
mysql_
connect()
变成pdo_
connect()
mysql_
query()
变成pdo_
query()
mysql_
num_rows()
变成pdo_
num_rows()
mysql_
insert_id()
变成pdo_
insert_id()
mysql_
fetch_array()
变成pdo_
fetch_array()
mysql_
fetch_assoc()
变成pdo_
fetch_assoc()
mysql_
real_escape_string()
变成pdo_
real_escape_string()
您的代码将同样工作,并且大部分看起来仍然相同:
include_once("pdo_mysql.php");
pdo_connect("localhost", "usrABC", "pw1234567");
pdo_select_db("test");
$result = pdo_query("SELECT title, html FROM pages");
while ($row = pdo_fetch_assoc($result)) {
print "$row[title] - $row[html]";
}
等等瞧。
您的代码正在使用PDO。
现在是时候实际使用它了。
您只需要一个不那么笨重的 API。
pdo_query()
为绑定参数添加了非常简单的支持。转换旧代码很简单:
将变量移出 SQL 字符串。
pdo_query()
.?
在变量之前的位置放置问号作为占位符。'
以前包含字符串值/变量的单引号。对于更长的代码,优势变得更加明显。
通常,字符串变量不仅被插入到 SQL 中,而且还与中间的转义调用连接起来。
pdo_query("SELECT id, links, html, title, user, date FROM articles
WHERE title='" . pdo_real_escape_string($title) . "' OR id='".
pdo_real_escape_string($title) . "' AND user <> '" .
pdo_real_escape_string($root) . "' ORDER BY date")
应用?
占位符后,您不必为此烦恼:
pdo_query("SELECT id, links, html, title, user, date FROM articles
WHERE title=? OR id=? AND user<>? ORDER BY date", $title, $id, $root)
请记住 pdo_* 仍然允许or。
只是不要转义变量并将其绑定在同一个查询中。
:named
占位符列表。更重要的是,您可以在任何查询后面安全地传递 $_REQUEST[] 变量。当提交<form>
的字段与数据库结构完全匹配时,它会更短:
pdo_query("INSERT INTO pages VALUES (?,?,?,?,?)", $_POST);
如此简单。但是,让我们回到一些关于为什么你可能想要摆脱和逃避的重写建议和技术原因。mysql_
sanitize()
功能将所有调用转换为mysql_
pdo_query
绑定参数后,删除所有冗余pdo_real_escape_string
调用。
特别是,您应该以一种或另一种形式修复过时的教程所宣传的任何sanitize
或clean
或filterThis
或功能:clean_data
function sanitize($str) {
return trim(strip_tags(htmlentities(pdo_real_escape_string($str))));
}
这里最明显的错误是缺乏文档。更重要的是,过滤顺序完全错误。
正确的顺序应该是:不推荐使用stripslashes
作为最内层调用,然后trim
是strip_tags
作为htmlentities
输出上下文的 ,最后是_escape_string
作为其应用程序应该直接在 SQL 交叉解析之前的调用。
但第一步只是摆脱_real_escape_string
电话。
sanitize()
如果您的数据库和应用程序流需要 HTML-context-safe 字符串,您现在可能必须保留其余的函数。添加一条注释,说明它今后仅适用 HTML 转义。
字符串/值处理委托给 PDO 及其参数化语句。
如果stripslashes()
在您的 sanitize 功能中提到任何内容,则可能表示更高级别的监督。
这通常是为了消除已弃用的magic_quotes
. 然而,最好集中固定,而不是一串一串。
使用一种用户态反转方法。然后删除函数stripslashes()
中的sanitize
。
关于magic_quotes 的历史记录。该功能已被正确弃用。然而,它经常被错误地描述为失败的安全功能。但是,magic_quotes 是一个失败的安全功能,就像网球作为营养来源失败一样。这根本不是他们的目的。
PHP2/FI 中的原始实现仅使用“引号将被自动转义,从而更容易将表单数据直接传递给 msql 查询”显式地引入它。值得注意的是,它意外地与mSQL一起使用是安全的,因为它仅支持 ASCII。
然后 PHP3/Zend 为 MySQL 重新引入了 magic_quotes 并错误地记录了它。但最初它只是一个便利功能,而不是为了安全。
当您将字符串变量打乱到 SQL 查询中时,它不仅会变得更加复杂,而且您需要遵循。MySQL 再次分离代码和数据也是多余的工作。
SQL 注入只是当数据渗入代码上下文时。数据库服务器以后无法发现 PHP 最初将变量粘合在查询子句之间的位置。
使用绑定参数,您可以在 PHP 代码中分隔 SQL 代码和 SQL 上下文值。但它不会在幕后再次被洗牌(PDO::EMULATE_PREPARES 除外)。您的数据库接收不变的 SQL 命令和 1:1 变量值。
虽然这个答案强调您应该关心 drop 的可读性优势。由于这种可见的技术数据/代码分离,有时还具有性能优势(重复的 INSERT 具有不同的值)。mysql_
请注意,参数绑定仍然不是针对所有SQL 注入的神奇的一站式解决方案。它处理数据/值的最常见用途。但不能将列名/表标识符列入白名单,帮助动态子句构造,或者只是简单的数组值列表。
这些pdo_*
包装函数构成了一个编码友好的权宜之计 API。(如果不是因为特殊的函数签名转变,这几乎是MYSQLI
可能的)。他们也经常暴露真实的 PDO。
重写不必停留在使用新的 pdo_ 函数名。您可以将每个 pdo_query() 一个一个转换为普通的 $pdo->prepare()->execute() 调用。
然而,最好从简化开始。例如常见的结果获取:
$result = pdo_query("SELECT * FROM tbl");
while ($row = pdo_fetch_assoc($result)) {
可以只用一个 foreach 迭代替换:
foreach ($result as $row) {
或者更好的是直接和完整的数组检索:
$result->fetchAll();
在大多数情况下,您会得到比 PDO 或 mysql_ 通常在查询失败后提供的更有用的警告。
所以这有希望可视化一些实际的原因和一个有价值的下降途径。mysql_
只是切换到pdo并不能完全削减它。pdo_query()
也只是它的前端。
除非您还引入参数绑定或可以利用更好的 API 中的其他内容,否则这是一个毫无意义的开关。我希望它被描述得足够简单,以免进一步打击新人。(教育通常比禁止更有效。)
虽然它符合最简单的可能工作的类别,但它仍然是非常实验性的代码。我是周末才写的。然而,有很多选择。只需谷歌搜索PHP 数据库抽象并浏览一下。一直存在并且将会有很多优秀的库来完成这些任务。
如果你想进一步简化你的数据库交互,像Paris/Idiorm这样的映射器值得一试。就像没有人在 JavaScript 中使用平淡无奇的 DOM 一样,现在您不必照看原始数据库接口。
mysql_
功能:
说起技术原因,只有少数,极其具体,很少使用。很可能你永远不会在你的生活中使用它们。
也许我太无知了,但我从来没有机会使用它们,比如
如果你需要它们——这些无疑是从 mysql 扩展转向更时尚和现代的技术原因。
尽管如此,也有一些非技术问题,这可能会使您的体验更加困难
后一个问题是个问题。
但是,在我看来,所提出的解决方案也好不到哪里去。
在我看来,所有这些 PHP 用户都将学习如何立即正确处理 SQL 查询的梦想太理想化了。他们很可能只是机械地将 mysql_* 更改为 mysqli_* ,使方法保持不变。特别是因为 mysqli 使准备好的语句的使用变得难以置信的痛苦和麻烦。
更不用说原生准备好的语句不足以防止SQL 注入,而且 mysqli 和 PDO 都没有提供解决方案。
因此,与其对抗这种诚实的扩展,我更愿意对抗错误的做法并以正确的方式教育人们。
此外,还有一些错误或无关紧要的原因,例如
mysql_query("CALL my_proc");
了很长时间)最后一个是一个有趣的点。尽管 mysql ext 不支持原生准备语句,但出于安全考虑,它们不是必需的。我们可以使用手动处理的占位符轻松伪造准备好的语句(就像 PDO 一样):
function paraQuery()
{
$args = func_get_args();
$query = array_shift($args);
$query = str_replace("%s","'%s'",$query);
foreach ($args as $key => $val)
{
$args[$key] = mysql_real_escape_string($val);
}
$query = vsprintf($query, $args);
$result = mysql_query($query);
if (!$result)
{
throw new Exception(mysql_error()." [$query]");
}
return $result;
}
$query = "SELECT * FROM table where a=%s AND b LIKE %s LIMIT %d";
$result = paraQuery($query, $a, "%$b%", $limit);
瞧,一切都是参数化且安全的。
不过好吧,如果你不喜欢手册里的红框,就出现了一个选择问题:mysqli还是PDO?
好吧,答案如下:
如果像绝大多数 PHP 人一样,您在应用程序代码中使用原始 API 调用(这本质上是错误的做法) - PDO 是唯一的选择,因为这个扩展伪装成不仅仅是 API,而是半 DAL,仍然不完整,但提供了许多重要的特性,其中两个使 PDO 与 mysqli 有严格的区别:
所以,如果你是一个普通的 PHP 用户,并且想在使用原生准备语句时省去很多麻烦,那么 PDO - 再次 - 是唯一的选择。
但是,PDO 也不是灵丹妙药,也有其困难。因此,我为PDO 标签 wiki
中的所有常见陷阱和复杂案例编写了解决方案
然而,每个谈论扩展的人总是忽略关于 Mysqli 和 PDO的两个重要事实:
准备好的陈述不是灵丹妙药。有些动态标识符无法使用准备好的语句绑定。存在参数数量未知的动态查询,这使得查询构建成为一项艰巨的任务。
mysqli_* 和 PDO 函数都不应该出现在应用程序代码中。
它们和应用程序代码之间应该有一个抽象层,它将完成内部绑定、循环、错误处理等所有脏活,使应用程序代码干燥和干净。特别是对于动态查询构建等复杂情况。
所以,仅仅切换到 PDO 或 mysqli 是不够的。必须使用 ORM、查询构建器或任何数据库抽象类,而不是在其代码中调用原始 API 函数。
相反——如果你的应用程序代码和mysql API之间有一个抽象层——使用哪个引擎实际上并不重要。您可以使用 mysql ext 直到它被弃用,然后轻松地将您的抽象类重写到另一个引擎,使所有应用程序代码保持不变。
下面是一些基于我的safemysql 类的例子,展示了这样一个抽象类应该是怎样的:
$city_ids = array(1,2,3);
$cities = $db->getCol("SELECT name FROM cities WHERE is IN(?a)", $city_ids);
将这一行与使用 PDO 所需的代码量进行比较。
然后与原始 Mysqli 准备语句所需的大量代码进行比较。请注意,错误处理、分析、查询日志记录已经内置并正在运行。
$insert = array('name' => 'John', 'surname' => "O'Hara");
$db->query("INSERT INTO users SET ?u", $insert);
将其与通常的 PDO 插入进行比较,在所有这些众多命名占位符、绑定和查询定义中,每个字段名称都重复六到十次。
另一个例子:
$data = $db->getAll("SELECT * FROM goods ORDER BY ?n", $_GET['order']);
您很难找到 PDO 处理此类实际案例的示例。
而且它会太罗嗦,而且很可能不安全。
所以,再一次——你关心的不仅仅是原始驱动程序,还有抽象类,它不仅对初学者手册中的愚蠢例子有用,而且对解决任何现实生活中的问题都有用。
原因有很多,但最重要的一个可能是这些函数鼓励不安全的编程实践,因为它们不支持预准备语句。准备好的语句有助于防止 SQL 注入攻击。
使用mysql_*
函数时,您必须记住通过mysql_real_escape_string()
. 如果您只忘记了一个地方,或者您碰巧只转义了部分输入,那么您的数据库可能会受到攻击。
PDO
在or中使用准备好的语句mysqli
会使得这些类型的编程错误更难发生。
因为(除其他原因外)确保输入数据得到清理要困难得多。如果您使用参数化查询,就像使用 PDO 或 mysqli 一样,您可以完全避免风险。
例如,有人可以用"enhzflep); drop table users"
作用户名。旧函数将允许每个查询执行多个语句,因此像那个讨厌的虫子这样的东西可以删除整个表。
如果要使用 mysqli 的 PDO,用户名最终将是"enhzflep); drop table users"
.
请参阅bobby-tables.com。
编写此答案是为了说明绕过编写不佳的 PHP 用户验证代码是多么微不足道,这些攻击如何(以及使用什么)起作用以及如何用安全的准备好的语句替换旧的 MySQL 函数 - 基本上,为什么 StackOverflow 用户(可能有很多代表)对新用户大吼大叫,他们提出问题以改进他们的代码。
首先,请随意创建这个测试 mysql 数据库(我已经调用了我的准备):
mysql> create table users(
-> id int(2) primary key auto_increment,
-> userid tinytext,
-> pass tinytext);
Query OK, 0 rows affected (0.05 sec)
mysql> insert into users values(null, 'Fluffeh', 'mypass');
Query OK, 1 row affected (0.04 sec)
mysql> create user 'prepared'@'localhost' identified by 'example';
Query OK, 0 rows affected (0.01 sec)
mysql> grant all privileges on prep.* to 'prepared'@'localhost' with grant option;
Query OK, 0 rows affected (0.00 sec)
完成后,我们可以转到我们的 PHP 代码。
让我们假设以下脚本是网站管理员的验证过程(简化但如果您复制并使用它进行测试,则可以正常工作):
<?php
if(!empty($_POST['user']))
{
$user=$_POST['user'];
}
else
{
$user='bob';
}
if(!empty($_POST['pass']))
{
$pass=$_POST['pass'];
}
else
{
$pass='bob';
}
$database='prep';
$link=mysql_connect('localhost', 'prepared', 'example');
mysql_select_db($database) or die( "Unable to select database");
$sql="select id, userid, pass from users where userid='$user' and pass='$pass'";
//echo $sql."<br><br>";
$result=mysql_query($sql);
$isAdmin=false;
while ($row = mysql_fetch_assoc($result)) {
echo "My id is ".$row['id']." and my username is ".$row['userid']." and lastly, my password is ".$row['pass']."<br>";
$isAdmin=true;
// We have correctly matched the Username and Password
// Lets give this person full access
}
if($isAdmin)
{
echo "The check passed. We have a verified admin!<br>";
}
else
{
echo "You could not be verified. Please try again...<br>";
}
mysql_close($link);
?>
<form name="exploited" method='post'>
User: <input type='text' name='user'><br>
Pass: <input type='text' name='pass'><br>
<input type='submit'>
</form>
乍一看似乎足够合法。
用户必须输入登录名和密码,对吗?
太棒了,不输入以下内容:
user: bob
pass: somePass
并提交。
输出如下:
You could not be verified. Please try again...
极好的!按预期工作,现在让我们尝试实际的用户名和密码:
user: Fluffeh
pass: mypass
惊人的!全面击掌,代码正确验证了管理员。这是完美的!
嗯,不是真的。可以说用户是一个聪明的小人物。可以说那个人是我。
输入以下内容:
user: bob
pass: n' or 1=1 or 'm=m
输出是:
The check passed. We have a verified admin!
恭喜,您只允许我进入您的超级保护管理员专区,而我输入了错误的用户名和错误的密码。说真的,如果你不相信我,用我提供的代码创建数据库,然后运行这个 PHP 代码 - 乍一看,它似乎确实很好地验证了用户名和密码。
所以,作为回答,这就是你被骂的原因。
那么,让我们看看出了什么问题,以及为什么我刚刚进入您的超级管理员专用蝙蝠洞。我猜测并假设您对输入不小心,只是将它们直接传递给数据库。我以一种会更改您实际运行的查询的方式构建输入。那么,它应该是什么,它最终是什么?
select id, userid, pass from users where userid='$user' and pass='$pass'
这就是查询,但是当我们用我们使用的实际输入替换变量时,我们得到以下信息:
select id, userid, pass from users where userid='bob' and pass='n' or 1=1 or 'm=m'
看看我是如何构造我的“密码”的,以便它首先关闭密码周围的单引号,然后引入一个全新的比较?然后为了安全起见,我添加了另一个“字符串”,以便单引号在我们最初拥有的代码中按预期关闭。
然而,这不是人们现在对你大喊大叫,而是向你展示如何让你的代码更安全。
好的,那么出了什么问题,我们该如何解决?
这是典型的 SQL 注入攻击。最简单的事情之一。在攻击向量的规模上,这是一个蹒跚学步的孩子攻击坦克 - 并获胜。
那么,我们如何保护您神圣的管理部分并使其变得美观和安全?要做的第一件事是停止使用那些非常古老和不推荐使用的mysql_*
功能。我知道,您遵循了您在网上找到的教程并且它有效,但它很旧,已经过时并且在几分钟的时间内,我刚刚突破它而没有流汗。
现在,您可以更好地选择使用mysqli_或PDO。我个人是 PDO 的忠实拥护者,因此我将在此答案的其余部分中使用 PDO。有优点也有缺点,但我个人觉得优点远大于缺点。它可以跨多个数据库引擎进行移植——无论您使用的是 MySQL 还是 Oracle 或任何该死的东西——只需更改连接字符串,它就具有我们想要使用的所有花哨的功能,而且它既漂亮又干净。我喜欢干净。
现在,让我们再看一下这段代码,这次是使用 PDO 对象编写的:
<?php
if(!empty($_POST['user']))
{
$user=$_POST['user'];
}
else
{
$user='bob';
}
if(!empty($_POST['pass']))
{
$pass=$_POST['pass'];
}
else
{
$pass='bob';
}
$isAdmin=false;
$database='prep';
$pdo=new PDO ('mysql:host=localhost;dbname=prep', 'prepared', 'example');
$sql="select id, userid, pass from users where userid=:user and pass=:password";
$myPDO = $pdo->prepare($sql, array(PDO::ATTR_CURSOR => PDO::CURSOR_FWDONLY));
if($myPDO->execute(array(':user' => $user, ':password' => $pass)))
{
while($row=$myPDO->fetch(PDO::FETCH_ASSOC))
{
echo "My id is ".$row['id']." and my username is ".$row['userid']." and lastly, my password is ".$row['pass']."<br>";
$isAdmin=true;
// We have correctly matched the Username and Password
// Lets give this person full access
}
}
if($isAdmin)
{
echo "The check passed. We have a verified admin!<br>";
}
else
{
echo "You could not be verified. Please try again...<br>";
}
?>
<form name="exploited" method='post'>
User: <input type='text' name='user'><br>
Pass: <input type='text' name='pass'><br>
<input type='submit'>
</form>
主要区别在于没有更多mysql_*
功能。这一切都是通过 PDO 对象完成的,其次,它使用的是准备好的语句。现在,你问的预先声明是什么?这是一种在运行查询之前告诉数据库的方法,即我们将要运行的查询是什么。在这种情况下,我们告诉数据库:“嗨,我将运行一个 select 语句,想要 id、userid 和 pass 从表 users 中 userid 是一个变量,而 pass 也是一个变量。”。
然后,在执行语句中,我们将一个数组传递给数据库,其中包含它现在期望的所有变量。
结果太棒了。让我们再次尝试以前的用户名和密码组合:
user: bob
pass: somePass
用户未通过验证。惊人的。
怎么样:
user: Fluffeh
pass: mypass
哦,我只是有点兴奋,它起作用了:支票通过了。我们有经过验证的管理员!
现在,让我们尝试一下聪明人会输入的数据,以尝试通过我们的小验证系统:
user: bob
pass: n' or 1=1 or 'm=m
这一次,我们得到以下信息:
You could not be verified. Please try again...
这就是为什么您在发布问题时被大喊大叫的原因 - 这是因为人们可以看到您的代码甚至可以在没有尝试的情况下被绕过。请务必使用此问题和答案来改进您的代码,使其更安全并使用最新的功能。
最后,这并不是说这是完美的代码。你可以做更多的事情来改进它,例如使用散列密码,确保当你在数据库中存储有意义的信息时,你不会以纯文本形式存储它,有多个级别的验证 - 但实际上,如果您只需将旧的易于注入的代码更改为此,您将在编写好代码的过程中做得很好 - 而且您已经走到这一步并且仍在阅读的事实让我有一种希望,您不仅会实现这种类型编写您的网站和应用程序时的代码,但您可能会出去研究我刚刚提到的其他内容 - 等等。写你能写的最好的代码,而不是勉强能用的最基本的代码。
我发现上面的答案真的很长,所以总结一下:
mysqli 扩展有很多好处,对 mysql 扩展的主要增强是:
- 面向对象的接口
- 支持准备好的报表
- 支持多个语句
- 交易支持
- 增强的调试功能
- 嵌入式服务器支持
资料来源:MySQLi 概述
如上述答案中所述,mysql 的替代品是 mysqli 和 PDO(PHP 数据对象)。
MySQLi 和 PDO 都是在 PHP 5.0 中引入的,而 MySQL 是在 PHP 3.0 之前引入的。需要注意的一点是 MySQL 包含在 PHP5.x 中,尽管在以后的版本中已弃用。
mysql_*
使用 mysqli 或 PDO几乎可以定义所有函数。只需将它们包含在您的旧 PHP 应用程序之上,它就可以在 PHP7 上运行。我的解决方案在这里。
<?php
define('MYSQL_LINK', 'dbl');
$GLOBALS[MYSQL_LINK] = null;
function mysql_link($link=null) {
return ($link === null) ? $GLOBALS[MYSQL_LINK] : $link;
}
function mysql_connect($host, $user, $pass) {
$GLOBALS[MYSQL_LINK] = mysqli_connect($host, $user, $pass);
return $GLOBALS[MYSQL_LINK];
}
function mysql_pconnect($host, $user, $pass) {
return mysql_connect($host, $user, $pass);
}
function mysql_select_db($db, $link=null) {
$link = mysql_link($link);
return mysqli_select_db($link, $db);
}
function mysql_close($link=null) {
$link = mysql_link($link);
return mysqli_close($link);
}
function mysql_error($link=null) {
$link = mysql_link($link);
return mysqli_error($link);
}
function mysql_errno($link=null) {
$link = mysql_link($link);
return mysqli_errno($link);
}
function mysql_ping($link=null) {
$link = mysql_link($link);
return mysqli_ping($link);
}
function mysql_stat($link=null) {
$link = mysql_link($link);
return mysqli_stat($link);
}
function mysql_affected_rows($link=null) {
$link = mysql_link($link);
return mysqli_affected_rows($link);
}
function mysql_client_encoding($link=null) {
$link = mysql_link($link);
return mysqli_character_set_name($link);
}
function mysql_thread_id($link=null) {
$link = mysql_link($link);
return mysqli_thread_id($link);
}
function mysql_escape_string($string) {
return mysql_real_escape_string($string);
}
function mysql_real_escape_string($string, $link=null) {
$link = mysql_link($link);
return mysqli_real_escape_string($link, $string);
}
function mysql_query($sql, $link=null) {
$link = mysql_link($link);
return mysqli_query($link, $sql);
}
function mysql_unbuffered_query($sql, $link=null) {
$link = mysql_link($link);
return mysqli_query($link, $sql, MYSQLI_USE_RESULT);
}
function mysql_set_charset($charset, $link=null){
$link = mysql_link($link);
return mysqli_set_charset($link, $charset);
}
function mysql_get_host_info($link=null) {
$link = mysql_link($link);
return mysqli_get_host_info($link);
}
function mysql_get_proto_info($link=null) {
$link = mysql_link($link);
return mysqli_get_proto_info($link);
}
function mysql_get_server_info($link=null) {
$link = mysql_link($link);
return mysqli_get_server_info($link);
}
function mysql_info($link=null) {
$link = mysql_link($link);
return mysqli_info($link);
}
function mysql_get_client_info() {
$link = mysql_link();
return mysqli_get_client_info($link);
}
function mysql_create_db($db, $link=null) {
$link = mysql_link($link);
$db = str_replace('`', '', mysqli_real_escape_string($link, $db));
return mysqli_query($link, "CREATE DATABASE `$db`");
}
function mysql_drop_db($db, $link=null) {
$link = mysql_link($link);
$db = str_replace('`', '', mysqli_real_escape_string($link, $db));
return mysqli_query($link, "DROP DATABASE `$db`");
}
function mysql_list_dbs($link=null) {
$link = mysql_link($link);
return mysqli_query($link, "SHOW DATABASES");
}
function mysql_list_fields($db, $table, $link=null) {
$link = mysql_link($link);
$db = str_replace('`', '', mysqli_real_escape_string($link, $db));
$table = str_replace('`', '', mysqli_real_escape_string($link, $table));
return mysqli_query($link, "SHOW COLUMNS FROM `$db`.`$table`");
}
function mysql_list_tables($db, $link=null) {
$link = mysql_link($link);
$db = str_replace('`', '', mysqli_real_escape_string($link, $db));
return mysqli_query($link, "SHOW TABLES FROM `$db`");
}
function mysql_db_query($db, $sql, $link=null) {
$link = mysql_link($link);
mysqli_select_db($link, $db);
return mysqli_query($link, $sql);
}
function mysql_fetch_row($qlink) {
return mysqli_fetch_row($qlink);
}
function mysql_fetch_assoc($qlink) {
return mysqli_fetch_assoc($qlink);
}
function mysql_fetch_array($qlink, $result=MYSQLI_BOTH) {
return mysqli_fetch_array($qlink, $result);
}
function mysql_fetch_lengths($qlink) {
return mysqli_fetch_lengths($qlink);
}
function mysql_insert_id($qlink) {
return mysqli_insert_id($qlink);
}
function mysql_num_rows($qlink) {
return mysqli_num_rows($qlink);
}
function mysql_num_fields($qlink) {
return mysqli_num_fields($qlink);
}
function mysql_data_seek($qlink, $row) {
return mysqli_data_seek($qlink, $row);
}
function mysql_field_seek($qlink, $offset) {
return mysqli_field_seek($qlink, $offset);
}
function mysql_fetch_object($qlink, $class="stdClass", array $params=null) {
return ($params === null)
? mysqli_fetch_object($qlink, $class)
: mysqli_fetch_object($qlink, $class, $params);
}
function mysql_db_name($qlink, $row, $field='Database') {
mysqli_data_seek($qlink, $row);
$db = mysqli_fetch_assoc($qlink);
return $db[$field];
}
function mysql_fetch_field($qlink, $offset=null) {
if ($offset !== null)
mysqli_field_seek($qlink, $offset);
return mysqli_fetch_field($qlink);
}
function mysql_result($qlink, $offset, $field=0) {
if ($offset !== null)
mysqli_field_seek($qlink, $offset);
$row = mysqli_fetch_array($qlink);
return (!is_array($row) || !isset($row[$field]))
? false
: $row[$field];
}
function mysql_field_len($qlink, $offset) {
$field = mysqli_fetch_field_direct($qlink, $offset);
return is_object($field) ? $field->length : false;
}
function mysql_field_name($qlink, $offset) {
$field = mysqli_fetch_field_direct($qlink, $offset);
if (!is_object($field))
return false;
return empty($field->orgname) ? $field->name : $field->orgname;
}
function mysql_field_table($qlink, $offset) {
$field = mysqli_fetch_field_direct($qlink, $offset);
if (!is_object($field))
return false;
return empty($field->orgtable) ? $field->table : $field->orgtable;
}
function mysql_field_type($qlink, $offset) {
$field = mysqli_fetch_field_direct($qlink, $offset);
return is_object($field) ? $field->type : false;
}
function mysql_free_result($qlink) {
try {
mysqli_free_result($qlink);
} catch (Exception $e) {
return false;
}
return true;
}
弃用的意思:
这意味着不要使用某些特定的功能/方法/软件功能/特定的软件实践,它只是意味着不应该使用它,因为应该使用该软件中(或将会有)更好的替代方案。
使用不推荐使用的函数时可能会出现几个常见问题:
1. 功能完全停止工作:应用程序或脚本可能依赖于不再支持的功能,因此使用它们的改进版本或替代方案。
2. 显示有关弃用的警告消息:这些消息通常不会干扰站点功能。但是,在某些情况下,它们可能会中断服务器发送标头的过程。
例如:这可能会导致登录问题(cookies/sessions 设置不正确)或转发问题(301/302/303 重定向)。
请记住:
- 弃用的软件仍然是软件的一部分。
- 弃用的代码只是代码的状态(标签)。
MYSQL 与 MYSQLI mysql*的主要区别
mysqli