使用 pimple 作为我的 DI 容器,我一直在勇敢地重构小类以依赖 DI 注入,消除了我可以看到很容易删除的硬编码依赖项。
我完成这项任务的方法非常简单,但我不知道它是否合适,因为除了上个月在这里学到的知识之外,我对 DI 和单元测试的经验很少。
我创建了一个类,ContainerFactory,它子类会疙瘩,并在该子类中创建了仅返回特定对象的容器的方法。
构造函数根据类型调用正确的创建者方法:
function __construct($type=null, $mode = null){
if(isset($type)){
switch ($type) {
case 'DataFactory':
$this->buildDataFactoryContainer($mode);
break;
case 'DbConnect':
$this->buildDbConnectContainer($mode);
break;
default:
return false;
}
}
}
容器对象创建的方法签名如下:
public function buildDataFactoryContainer($mode=null)
这个想法是,我可以在调用此容器时设置$mode进行测试,并让它加载测试值而不是实际的运行时设置。我想避免编写单独的容器类进行测试,这是我认为没有测试相关代码的简单方法。
相反,我可以对 ContainerFactory 进行子类化,即: ContainerFactoryTesting extends ContainerFactory
并覆盖其中,而不是将测试代码与应用程序代码混合在一起,并使用 $mode=null 将方法签名混杂在一起,但这不是本文的重点。 继续,要为特定对象创建容器,我只需这样做:
// returns container with DataFactory dependencies, holds $db and $logger objects.
$dataFactoryContainer = new ContainerFactory('DataFactory');
// returns container with test settings.
$dataFactoryTestContainer = new ContainerFactory('DataFactory','test');
// returns container with DbConnect dependencies, holds dbconfig and $logger objects.
$dbConnectContainer = new ContainerFactory('DbConnect');
我刚刚遇到了一个问题,让我怀疑我正在建立的策略是有缺陷的。
从上面看,数据工厂包含$db保存数据库连接的对象。我现在正在重构这个数据库类以删除它对$registry对象的依赖,但是,当我添加需要$dbConnectContainer的对象时$db我将如何创建$dataFactoryContainer?
例如,在数据工厂容器中,我添加了一个 dbconnect 实例,但 IT 现在需要一个传递给它的容器......
我意识到我的英语不是那么好,希望我已经解释得足够好,让SO'er同事理解。
我的问题是双重的,你们如何处理以简单的方式为本身包含依赖项的依赖项创建对象?
以及..您如何分离容器配置以创建用于测试目的的对象?
与往常一样,任何评论或相关帖子的链接都值得赞赏。
不应为所有内容创建单独的容器,而应使用单个容器。您可以创建一个容器"扩展",它基本上只是在您的容器上设置服务。
这是我建议的设置的一个广泛示例。当您有一个容器时,递归解析依赖项是微不足道的。如您所见,security.authentication_provider
取决于 db
,这取决于db.config
.
由于闭包的懒惰性,很容易定义服务,然后稍后定义它们的配置。此外,覆盖它们也很容易,正如您在TestExtension中看到的那样。
可以将服务定义拆分为多个单独的扩展,以使它们更易于重用。这几乎就是Silex微框架所做的(它使用pimple(。
我希望这能回答你的问题。
扩展接口
class ExtensionInterface
{
function register(Pimple $container);
}
应用扩展
class AppExtension
{
public function register(Pimple $container)
{
$container['mailer'] = $container->share(function ($container) {
return new Mailer($container['mailer.config']);
});
$container['db'] = $container->share(function ($container) {
return new DatabaseConnection($container['db.config']);
});
$container['security.authentication_provider'] = $container->share(function ($container) {
return new DatabaseAuthenticationProvider($container['db']);
});
}
}
配置扩展
class ConfigExtension
{
private $config;
public function __construct($configFile)
{
$this->config = json_decode(file_get_contents($configFile), true);
}
public function register(Pimple $container)
{
foreach ($this->config as $name => $value) {
$container[$name] = $container->protect($value);
}
}
}
config.json
{
"mailer.config": {
"username": "something",
"password": "secret",
"method": "smtp"
},
"db.config": {
"username": "root",
"password": "secret"
}
}
测试扩展
class TestExtension
{
public function register(Pimple $container)
{
$container['mailer'] = function () {
return new MockMailer();
};
}
}
集装箱工厂
class ContainerFactory
{
public function create($configDir)
{
$container = new Pimple();
$extension = new AppExtension();
$extension->register($container);
$extension = new ConfigExtension($configDir.'/config.json');
$extension->register($container);
return $container;
}
}
应用
$factory = new ContainerFactory();
$container = $factory->create();
测试引导程序
$factory = new ContainerFactory();
$container = $factory->create();
$extension = new TestExtension();
$extension->register($container);