
那些年,我们被依赖管理折磨的日子
还记得吗?当你开发一个稍微复杂一点的PHP应用时,代码中充斥着大量的 new SomeClass(),然后你需要手动将这些对象作为参数传递给其他类。例如,你的 OrderService 需要 ProductRepository 和 UserRepository,而 ProductRepository 又依赖 DatabaseAdapter。于是,你的代码变成了这样:
// 假设这是你手动创建依赖的方式 $dbAdapter = new DatabaseAdapter(); $productRepo = new ProductRepository($dbAdapter); $userRepo = new UserRepository($dbAdapter); $orderService = new OrderService($productRepo, $userRepo); $orderService->createOrder(...);
登录后复制
这看起来似乎没问题,但在实际项目中,一旦 DatabaseAdapter 的构造函数变了,或者你需要切换到另一个数据库实现,你可能需要修改所有依赖它的地方。更糟糕的是,当你想对 OrderService 进行单元测试时,你需要手动模拟 ProductRepository 和 UserRepository,这使得测试变得异常复杂和脆弱。
这种手动管理依赖的方式,不仅导致代码高度耦合,难以维护和扩展,还让单元测试成为一场噩梦,最终形成难以阅读和理解的“意大利面条式代码”。我们急需一种更优雅、更高效的方式来管理对象及其依赖。
Composer 登场:引入 topthink/think-container 解决之道
幸运的是,PHP社区拥有强大的工具——Composer,它让引入和管理外部库变得轻而易举。而 topthink/think-container 正是解决上述依赖管理困境的利器。它是一个遵循 PSR-11 规范的依赖注入(DI)容器和门面(Facade)管理器,能帮助我们彻底告别手动依赖管理的烦恼。
立即学习“PHP免费学习笔记(深入)”;
首先,通过 Composer 简单地安装它:
composer require topthink/think-container
登录后复制
安装完成后,我们就可以开始使用 think-container 来重构我们的代码了。
依赖注入的魔力:让容器帮你管理对象
topthink/think-container 的核心是依赖注入。它允许你将类的创建和依赖关系交给容器来处理。当你需要一个对象时,你不是手动 new 它,而是向容器“索取”,容器会负责解析并提供所有必要的依赖。
让我们看看如何使用它:
use think\Container;
// 假设我们有这些类
class DatabaseAdapter { /* ... */ }
class ProductRepository {
public function __construct(DatabaseAdapter $adapter) { /* ... */ }
}
class UserRepository {
public function __construct(DatabaseAdapter $adapter) { /* ... */ }
}
class OrderService {
public function __construct(ProductRepository $productRepo, UserRepository $userRepo) { /* ... */ }
}
$container = Container::getInstance();
// 1. 绑定类到容器
// 容器会自动解析 DatabaseAdapter 的依赖(如果它有的话,这里没有)
$container->bind(DatabaseAdapter::class);
$container->bind(ProductRepository::class);
$container->bind(UserRepository::class);
$container->bind(OrderService::class);
// 2. 从容器中获取对象
// 容器会自动实例化 OrderService 并注入 ProductRepository 和 UserRepository,
// 而这两个又会自动注入 DatabaseAdapter
$orderService = $container->make(OrderService::class);
$orderService->createOrder(...);
// 你也可以绑定接口到具体实现,实现更好的解耦
interface CacheInterface { public function get($key); }
class FileCache implements CacheInterface { public function get($key) { /* ... */ } }
$container->bind(CacheInterface::class, FileCache::class);
$cache = $container->make(CacheInterface::class); // 得到 FileCache 实例登录后复制
通过 bind() 方法,我们告诉容器如何构建一个类。当使用 make() 方法时,容器会智能地解析构造函数中的类型提示,并自动注入所需的依赖。这极大地减少了代码中的 new 操作,让依赖关系变得清晰。
标签: composer php cad app 工具 ai
还木有评论哦,快来抢沙发吧~