
在php面向对象编程中,频繁地在每个方法中创建新的pdo数据库连接会导致资源浪费和代码冗余。本教程将介绍如何通过在类的构造函数中一次性创建pdo连接,并将其存储为类属性,从而实现连接的复用。通过这种方式,不仅能提高代码效率和可维护性,还能确保数据库资源被有效管理,避免不必要的连接开销。
在开发PHP应用程序时,尤其是采用面向对象(OOP)范式时,数据库连接是不可或缺的一部分。然而,初学者常犯的一个错误是在每个需要与数据库交互的方法内部都重新实例化一个PDO对象,例如:
protected function insertUser(){
$connectionvar = new PDO('mysql:host='. $this->host .';dbname='.$this->db, $this->user, $this->password);
// 后续的SQL执行代码
}这种做法存在显著问题:
解决上述问题的关键在于,只创建一次数据库连接,并将其存储在类的属性中,以便在整个对象生命周期中复用。 最常见且推荐的做法是在类的构造函数中初始化PDO连接,并将其赋值给一个受保护的(protected)或私有的(private)类属性。
以下是一个示例,展示如何构建一个基础的数据库连接类:
立即学习“PHP免费学习笔记(深入)”;
<?php
class DB
{
/**
* @var PDO 数据库连接实例
*/
protected PDO $connection;
/**
* 构造函数,在对象创建时建立数据库连接
*
* @param string $host 数据库主机名
* @param string $db 数据库名称
* @param string $user 数据库用户名
* @param string $password 数据库密码
*/
public function __construct(string $host, string $db, string $user, string $password)
{
try {
$dsn = "mysql:host={$host};dbname={$db};charset=utf8mb4";
$options = [
PDO::ATTR_ERRMODE => PDO::ERRMODE_EXCEPTION, // 错误模式:抛出异常
PDO::ATTR_DEFAULT_FETCH_MODE => PDO::FETCH_ASSOC, // 默认获取模式:关联数组
PDO::ATTR_EMULATE_PREPARES => false, // 禁用模拟预处理语句
];
$this->connection = new PDO($dsn, $user, $password, $options);
} catch (PDOException $e) {
// 在生产环境中,应记录错误而非直接输出
throw new PDOException("数据库连接失败: " . $e->getMessage(), (int)$e->getCode());
}
}
}在这个DB类中:
一旦PDO连接被存储为类属性,任何继承自DB类或通过依赖注入获得DB实例的类,都可以在其方法中通过$this->connection来访问这个已建立的连接,而无需再次实例化。
例如,如果你有一个用户管理类UserManager,它需要进行CRUD操作:
<?php
class UserManager extends DB
{
/**
* 插入新用户到数据库
*
* @param string $username 用户名
* @param string $email 邮箱
* @param string $password 密码(应加密存储)
* @return bool 插入是否成功
*/
public function insertUser(string $username, string $email, string $password): bool
{
// 直接使用父类中已建立的数据库连接
$stmt = $this->connection->prepare("INSERT INTO users (username, email, password) VALUES (?, ?, ?)");
return $stmt->execute([$username, $email, $password]);
}
/**
* 根据用户ID获取用户信息
*
* @param int $userId 用户ID
* @return array|false 用户信息数组或false
*/
public function getUserById(int $userId): array|false
{
$stmt = $this->connection->prepare("SELECT id, username, email FROM users WHERE id = ?");
$stmt->execute([$userId]);
return $stmt->fetch(); // 默认PDO::FETCH_ASSOC
}
// 其他CRUD方法...
}
// 示例使用
$dbHost = 'localhost';
$dbName = 'test_db';
$dbUser = 'root';
$dbPass = 'password';
try {
$userManager = new UserManager($dbHost, $dbName, $dbUser, $dbPass);
$isInserted = $userManager->insertUser('john_doe', 'john@example.com', 'hashed_password');
if ($isInserted) {
echo "用户插入成功!\n";
}
$user = $userManager->getUserById(1);
if ($user) {
echo "获取用户ID为1的信息:\n";
print_r($user);
} else {
echo "未找到用户ID为1的信息。\n";
}
} catch (PDOException $e) {
echo "操作失败: " . $e->getMessage() . "\n";
}
?>通过这种方式,UserManager类及其所有方法都可以共享同一个数据库连接,避免了重复实例化PDO对象。
为了进一步解耦和提高代码的复用性,最佳实践是让DB类不仅负责连接管理,还提供通用的数据库查询执行方法。这样,其他业务逻辑类(如UserManager)就不需要直接操作prepare、execute等方法,而是调用DB类提供的更高级别的接口。
修改DB类,添加一个通用的查询方法:
<?php
class DB
{
protected PDO $connection;
public function __construct(string $host, string $db, string $user, string $password)
{
try {
$dsn = "mysql:host={$host};dbname={$db};charset=utf8mb4";
$options = [
PDO::ATTR_ERRMODE => PDO::ERRMODE_EXCEPTION,
PDO::ATTR_DEFAULT_FETCH_MODE => PDO::FETCH_ASSOC,
PDO::ATTR_EMULATE_PREPARES => false,
];
$this->connection = new PDO($dsn, $user, $password, $options);
} catch (PDOException $e) {
throw new PDOException("数据库连接失败: " . $e->getMessage(), (int)$e->getCode());
}
}
/**
* 执行一个SQL查询并返回所有结果(适用于SELECT)
*
* @param string $sql SQL查询语句
* @param array $parameters 绑定到SQL语句的参数数组
* @return array 查询结果集
*/
public function runQueryAndGetResult(string $sql, array $parameters = []): array
{
$statement = $this->connection->prepare($sql);
$statement->execute($parameters);
return $statement->fetchAll();
}
/**
* 执行一个SQL查询并返回单行结果(适用于SELECT单行)
*
* @param string $sql SQL查询语句
* @param array $parameters 绑定到SQL语句的参数数组
* @return array|false 单行结果数组或false
*/
public function runQueryAndGetSingleResult(string $sql, array $parameters = []): array|false
{
$statement = $this->connection->prepare($sql);
$statement->execute($parameters);
return $statement->fetch();
}
/**
* 执行一个SQL语句并返回受影响的行数(适用于INSERT, UPDATE, DELETE)
*
* @param string $sql SQL语句
* @param array $parameters 绑定到SQL语句的参数数组
* @return int 受影响的行数
*/
public function runStatement(string $sql, array $parameters = []): int
{
$statement = $this->connection->prepare($sql);
$statement->execute($parameters);
return $statement->rowCount();
}
/**
* 获取最后插入的ID
*
* @return string|false
*/
public function getLastInsertId(): string|false
{
return $this->connection->lastInsertId();
}
}现在,UserManager类可以不再继承DB类,而是通过依赖注入的方式接收DB实例。这是一种更灵活、更推荐的实践,因为它降低了类之间的耦合度。
<?php
// 假设 DB 类已定义如上
class UserManager
{
private DB $db; // 持有 DB 类的实例
/**
* 构造函数,通过依赖注入接收 DB 实例
*
* @param DB $db 数据库操作对象
*/
public function __construct(DB $db)
{
$this->db = $db;
}
public function insertUser(string $username, string $email, string $password): bool
{
$sql = "INSERT INTO users (username, email, password) VALUES (?, ?, ?)";
$affectedRows = $this->db->runStatement($sql, [$username, $email, $password]);
return $affectedRows > 0;
}
public function getUserById(int $userId): array|false
{
$sql = "SELECT id, username, email FROM users WHERE id = ?";
return $this->db->runQueryAndGetSingleResult($sql, [$userId]);
}
public function getAllUsers(): array
{
$sql = "SELECT id, username, email FROM users";
return $this->db->runQueryAndGetResult($sql);
}
// 其他CRUD方法...
}
// 示例使用
$dbHost = 'localhost';
$dbName = 'test_db';
$dbUser = 'root';
$dbPass = 'password';
try {
$db = new DB($dbHost, $dbName, $dbUser, $dbPass); // 只创建一次 DB 实例
$userManager = new UserManager($db); // 将 DB 实例注入 UserManager
$isInserted = $userManager->insertUser('jane_doe', 'jane@example.com', 'another_hashed_password');
if ($isInserted) {
echo "用户 Jane Doe 插入成功!\n";
}
$user = $userManager->getUserById(2);
if ($user) {
echo "获取用户ID为2的信息:\n";
print_r($user);
} else {
echo "未找到用户ID为2的信息。\n";
}
$allUsers = $userManager->getAllUsers();
echo "所有用户信息:\n";
print_r($allUsers);
} catch (PDOException $e) {
echo "操作失败: " . $e->getMessage() . "\n";
}
?>通过这种分层设计:
通过在类的构造函数中一次性创建并存储PDO数据库连接为类属性,可以有效避免重复创建连接带来的性能问题和代码冗余。进一步地,将数据库操作封装在一个专门的DB类中,并通过依赖注入的方式将其提供给其他业务逻辑类,是实现高内聚、低耦合、可维护和可测试代码的最佳实践。
注意事项:
以上就是PHP面向对象编程中避免重复创建PDO数据库连接的最佳实践的详细内容,更多请关注php中文网其它相关文章!
PHP怎么学习?PHP怎么入门?PHP在哪学?PHP怎么学才快?不用担心,这里为大家提供了PHP速学教程(入门到精通),有需要的小伙伴保存下载就能学习啦!
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号