Zend framework2 ZF2:验证程序中的ServiceLocator无法获取实例

Zend framework2 ZF2:验证程序中的ServiceLocator无法获取实例,zend-framework2,Zend Framework2,我实现了一个自定义验证器类,如下所示 class UsernameUnique extends AbstractValidator implements ServiceLocatorAwareInterface { use ServiceLocatorAwareTrait; protected $userManager; public function getUserManager() { if(!$this->userManager)

我实现了一个自定义验证器类,如下所示

class UsernameUnique extends AbstractValidator implements ServiceLocatorAwareInterface
{
    use ServiceLocatorAwareTrait;

    protected $userManager;

    public function getUserManager()
    {
        if(!$this->userManager)
        {
            $this->userManager = $this->getServiceLocator()->get('RTUserManager');
        }
        return $this->userManager;
    }

    public isValid()
    {
        $um = $this->getUserManager();
        //rest of code
    }
}
我在表单中使用了这个验证器

'validators' => array(
    array(
        'name'  => 'RTUser\Form\Validator\EmailUnique'
         )
    )
一切看起来都很好,但是当表单验证发生时,我得到一个异常

Zend\Validator\ValidatorPluginManager::get was unable to fetch or create an instance for RTUserManager
堆栈跟踪指向Validator类

可以从控制器创建“RTUserManager”。因此,服务管理器配置没有问题。但是验证程序无法创建其实例

我怎样才能解决这个问题

编辑: 这是控制器插件中的代码

public function register()
{
    $newuser = $this->getInputValues();

    if($newuser->isValid())
    {
        $userdata = $newuser->getData(FormInterface::VALUES_AS_ARRAY);
        if($this->saveUserToDB($userdata))
        {
            return $message('Registration has been successful. Verify your email address to login.');
        }
    }
    return $this->registrationForm($newuser);
}

protected function getInputValues()
{
    $userdata = $this->getController()->getRequest()->getPost();
    $newuser = new RegistrationForm();
    $newuser->bind($userdata);
    return $newuser;
}
表单代码

class RegistrationForm extends Form
{
    protected $inputfilter;

    public function __construct() 
    {
        //code to add form elements
        $this->setInputFilter($this->getInputFilter());
    }

    public function getInputFilter() 
    {

       if(!$this->inputfilter)
       {
           $inputFilter = new InputFilter();
           //code to add other validators and filters

           $inputFilter->add(
               'name' => 'email',
               'validators' => array(
                    'name'  => 'RTUser\Form\Validator\EmailUnique',
               )
           );
           $this->inputfilter = $inputFilter;
       }
       return $this->inputfilter;
    }
}

Zend框架有多个管理器,它们都是服务定位器。如果您只知道
ServiceManager

在验证器中,
getServiceLocator
不会返回
ServiceManager
,而是返回
ValidatorPluginManager
,它当然不知道您的
服务管理器的配置。
幸运的是,它扩展了
AbstractPluginManager
——这意味着它包含实际的服务管理器

因此,在获得服务之前,您需要先访问
ServiceManager
。相反:

$this->userManager = $this->getServiceLocator()->get('RTUserManager');
尝试:

这只是一个更清晰的符号:

$this->userManager = $this->getServiceLocator()->getServiceLocator()->get('RTUserManager');
最后,我建议您编写一些factory for you validator来注入依赖项
RTUserManager
,以避免失去对依赖项的跟踪。在类中使用任何服务定位器都不是好的做法,在Zend 3中,服务定位器甚至可能只在工厂中可用。但现在这是可选的

编辑:

您是对的,直接创建
InputFilter
是您的第二个问题。因为您这样做,所以它不能了解
ServiceManager
。要解决此问题,您需要:

  • InputFilterPluginManager
    注入
    Zend\InputFilter\Factory
  • 使用此工厂创建
    InputFilter
  • 因此,您的
    注册表单应该如下所示:

    class RegistrationForm extends Form
    {
        protected $inputfilter;
        /**
         * @var InputFilterPluginManager
         */
        protected $inputFilterManager;
    
        /**
         * RegistrationForm constructor.
         * @param InputFilterPluginManager $inputFilterManager
         */
        public function __construct(InputFilterPluginManager $inputFilterManager)
        {
            $this->setInputFilterManager($inputFilterManager);
            parent::__construct("registration");
            //code to add form elements
            $this->setInputFilter($this->getInputFilter());
        }
    
        public function getInputFilter()
        {
            if(!$this->inputfilter)
            {
                $factory = new Factory($this->getInputFilterManager());
                $inputFilter = $factory->createInputFilter(array(
                    // other input specifications
                    "email" => array(
                        'name'       => 'email',
                        'validators' => array(
                            array(
                                'name' => 'Application\Form\Validator\EmailUnique',
                            ),
                        ),
                    ),
                ));
                $this->inputfilter = $inputFilter;
            }
            return $this->inputfilter;
        }
    
        /**
         * @return InputFilterPluginManager
         */
        public function getInputFilterManager()
        {
            return $this->inputFilterManager;
        }
    
        /**
         * @param InputFilterPluginManager $inputFilterManager
         */
        public function setInputFilterManager($inputFilterManager)
        {
            $this->inputFilterManager = $inputFilterManager;
        }
    
    }
    
    在实例化
    注册表单时,您需要直接传递
    InputFilterPluginManager
    ,或者为其构建工厂。为了简单起见,我将扩展您当前的插件代码,尽管我更喜欢工厂:

    protected function getInputValues()
    {
        /** @var InputFilterPluginManager $inputManager */
        $inputManager = $this->getController()->getServiceLocator()->get("InputFilterManager");
        $userdata = $this->getController()->getRequest()->getPost();
        $newuser = new RegistrationForm($inputManager);
        $newuser->bind($userdata);
        return $newuser;
    }
    

    就这样。工厂可以通过
    InputFilterPluginManager
    访问所有管理器,因此可以使用适当的依赖注入创建验证器。

    我已经尝试过了。但是
    $this->getServiceLocator()->getServiceLocator()返回空值。这可能是我使用
    new
    关键字创建表单实例的原因。创建验证器实例?如果是,可能是,请在创建验证器的位置添加代码。否。验证程序由zend隐式实例化。仅使用
    new
    关键字创建表单。我已编辑问题以添加代码。我还尝试从服务创建输入筛选器。但它给出了同样的结果。这只需要一个小小的改变。在表单中,在调用
    $this->setInputFilter()
    之前必须调用父构造函数。否则,将引发对null上的成员函数insert()的调用。如果你在回答中改变这一点,可能对其他人有用。非常感谢你。
    protected function getInputValues()
    {
        /** @var InputFilterPluginManager $inputManager */
        $inputManager = $this->getController()->getServiceLocator()->get("InputFilterManager");
        $userdata = $this->getController()->getRequest()->getPost();
        $newuser = new RegistrationForm($inputManager);
        $newuser->bind($userdata);
        return $newuser;
    }