PDO 和 MySQLi 是 PHP 中用于连接和操作 MySQL 数据库的两种主要扩展。它们各有优缺点,适用于不同的场景。以下是它们的详细对比:
1. 基本概念
PDO(PHP Data Objects):
是一个数据库访问抽象层,支持多种数据库(如 MySQL、PostgreSQL、SQLite 等)。
提供一致的 API,可以在不同数据库之间切换,而无需重写大量代码。
MySQLi(MySQL Improved):
是 MySQL 的专用扩展,仅支持 MySQL 数据库。
提供了面向对象和面向过程两种编程接口。
2. 连接方式
PDO 连接示例
<?php
$host = 'localhost';
$dbname = 'blog_system';
$username = 'root';
$password = '020123';
try {
$conn = new PDO("mysql:host=$host;dbname=$dbname", $username, $password);
$conn->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
echo "PDO 连接成功!";
} catch (PDOException $e) {
die("PDO 连接失败: " . $e->getMessage());
}
?>
MySQLi 连接示例
<?php
$host = 'localhost';
$username = 'root';
$password = '020123';
$dbname = 'blog_system';
$conn = new mysqli($host, $username, $password, $dbname);
if ($conn->connect_error) {
die("MySQLi 连接失败: " . $conn->connect_error);
}
echo "MySQLi 连接成功!";
?>
3. 主要区别
特性 PDO MySQLi
数据库支持 支持多种数据库(如 MySQL、PostgreSQL 等) 仅支持 MySQL
API 风格 仅面向对象 支持面向对象和面向过程
预处理语句 支持命名占位符(如 :name) 仅支持位置占位符(如 ?)
错误处理 支持异常处理(PDOException) 支持异常处理(需要手动启用)
性能 稍慢 稍快
功能丰富性 功能较少,但足够通用 功能更丰富(如异步查询、多语句执行等)
学习曲线 较简单 较复杂
4. 预处理语句
PDO 预处理语句
$sql = "SELECT * FROM users WHERE id = :id";
$stmt = $conn->prepare($sql);
$stmt->execute(['id' => 1]);
$user = $stmt->fetch();
MySQLi 预处理语句
$sql = "SELECT * FROM users WHERE id = ?";
$stmt = $conn->prepare($sql);
$stmt->bind_param("i", 1); // "i" 表示整数类型
$stmt->execute();
$result = $stmt->get_result();
$user = $result->fetch_assoc();
5. 错误处理
PDO 错误处理
PDO 默认使用静默模式,需要通过 setAttribute 启用异常模式:
$conn->setAttribute(PDO::ATTR_ERRMODE, PDO::ERRMODE_EXCEPTION);
MySQLi 错误处理
MySQLi 默认抛出错误,可以通过 mysqli_report 启用异常模式:
mysqli_report(MYSQLI_REPORT_ERROR | MYSQLI_REPORT_STRICT);
6. 事务处理
PDO 事务处理
$conn->beginTransaction();
try {
$conn->exec("INSERT INTO users (username, email) VALUES ('Alice', 'alice@example.com')");
$conn->exec("UPDATE accounts SET balance = balance - 100 WHERE user_id = 1");
$conn->commit();
} catch (PDOException $e) {
$conn->rollBack();
echo "事务失败: " . $e->getMessage();
}
MySQLi 事务处理
$conn->begin_transaction();
try {
$conn->query("INSERT INTO users (username, email) VALUES ('Alice', 'alice@example.com')");
$conn->query("UPDATE accounts SET balance = balance - 100 WHERE user_id = 1");
$conn->commit();
} catch (mysqli_sql_exception $e) {
$conn->rollback();
echo "事务失败: " . $e->getMessage();
}
7. 选择建议
选择 PDO 的场景:
需要支持多种数据库。
希望代码具有更好的可移植性。
喜欢简洁一致的 API。
选择 MySQLi 的场景:
仅使用 MySQL 数据库。
需要 MySQL 特有的功能(如异步查询、多语句执行等)。
对性能有较高要求。
总结
如果你需要跨数据库兼容性,或者希望代码更简洁,推荐使用 PDO。
如果你只使用 MySQL,并且需要更丰富的功能,推荐使用 MySQLi。
评论