我有一些传统课程。许多类是使用工厂类实例化的。
还有一个单例类。
将来我想用 DIC 完全替换它们。目前,代码库很大,无法做到这一点。
现在我的目标是将 DI-Container 注入到由 Singleton 类实例化的每个服务中。Singleton 类有一个带有此签名的静态方法。
final class Singleton
{
private static $singletonCache = array();
public static function getInstance($namespace, $className)
{
}
}
Run Code Online (Sandbox Code Playgroud)
在这个函数内部,我想检查:
$instance = new $className();
if($instance instanceof ContainerAwareInterface)
{
// TODO: how do we get the container here
$instance->setContainer($container);
}
Run Code Online (Sandbox Code Playgroud)
但是我怎样才能最好地将容器放入我的“单例类”中,它只是静态调用的?
另一种方法是在需要时全局访问容器:
public static function getInstance($namespace, $className)
{
$container = $_GLOBAL['kernel']->getContainer();
}
Run Code Online (Sandbox Code Playgroud)
当然,这种方法有一些问题,但只要你正在过渡,就足够了。
在引导代码早期的某个位置,但在实例化容器之后,您可以将容器传递给单例类:
Singleton::setContainer($container);
Run Code Online (Sandbox Code Playgroud)
它将容器存储在静态属性中:
final class Singleton
{
// ...
private static $container;
public static function setContainer(ContainerInterface $container)
{
self::$container = $container;
}
}
Run Code Online (Sandbox Code Playgroud)
然而,正如您在单例类的示例中了解到的那样,所有全局状态都会让您头疼。传递容器(并使用 ContainerAware)是需要避免的事情。通过将容器传递给您的服务,您使它们依赖于整个服务世界。仅传递您真正需要的协作者会更干净。它也更容易测试。