我有一个MVC应用程序,其中域模型定义良好,包含实体,存储库和服务层。
为了避免在我的控制器中实例化我的服务类,因此,使用不适合他们的逻辑弄乱我的控制器,我创建了一个帮助器,作为一种Service Locator,但在阅读了一下之后,我意识到很多开发者:
假设服务定位器实际上是反模式。但我认为我的实施不是一种反模式。
他们认为服务定位器是反模式的原因是因为它隐藏了依赖关系,但是,我注入唯一的依赖项(实体管理器,这种依赖关系可能不会改变,因为它在服务的签名中)在实例化服务定位器时,服务类所需的接口。
这是我的代码:
<?php
namespace App\Controller\Action\Helper;
use Zend_Controller_Action_Helper_Abstract as Helper,
Doctrine\ORM\EntityManager;
/**
* Service Locator Helper
* @author JCM
*/
class Service extends Helper {
/**
* The actual EntityManager
* @var \Doctrine\ORM\EntityManager
*/
private $entityManager;
/**
* Services Namespace
* @var string
*/
private $ns;
/**
* @param EntityManager $entityManager
* @param string $ns The namespace where to find the services
*/
public function __construct( EntityManager $entityManager, $ns )
{
$this->entityManager = $entityManager;
$this->ns = $ns;
}
/**
* @param string $serviceName
* @param array $options
* @param string $ns
*/
public function direct( $serviceName, array $options = array(), $ns = null )
{
$ns = ( (!$ns) ? $this->ns : $ns ) . '\\';
$class = $ns . $serviceName;
return new $class( $this->entityManager, $options );
}
/**
* @param EntityManager $entityManager
*/
public function setEntityManager( EntityManager $entityManager )
{
$this->entityManager = $entityManager;
}
/**
* @return \Doctrine\ORM\EntityManager
*/
public function getEntityManager()
{
return $this->entityManager;
}
/**
* @param string $name
*/
public function __get( $name )
{
return $this->direct( $name );
}
}
使用前端控制器注册Action Helper:
//inside some method in the bootstrap
HelperBroker::addHelper( new App\Controller\Action\Helper\Service( $entityManager, '\App\Domain\Service' ) );
我如何在我的控制器中使用这个助手:
//Some Controller
$myService = $this->_helper->service( 'MyService' ); //returns an instance of the class App\Domain\Service\MyService
$result = $myService->doSomethingWithSomeData( $this->getRequest()->getPost() );
//etc...
答案 0 :(得分:4)
我认为您构建的内容不会实现服务定位器模式。如果你有这样的话,某个地方会有一个全球性的“注册表”。
我看到的基本上是具有依赖关系的工厂类(App\Controller\Action\Helper\Service
),它们通过ctor注入。所以你的类不知道它的依赖来自哪里,也不负责创建它们(这是一件好事!)。
如果我错了,请纠正我。 :)
BTW:这也是你不应该传递依赖注入容器的原因。它变成了服务定位器。