Symfony 3:无法从控制器内部访问容器

时间:2017-06-15 15:08:12

标签: php symfony symfony-3.3

我将我的应用从Symfony 2.8迁移到Symfony 3.3。

从我的控制器内部我有这个:

public function indexAction()
{
    $email = new Email();

    $form = $this->createForm(GetStartedType::class, $email, [
        'action' => $this->generateUrl('get_started_end'),
        'method' => 'POST',
    ]);

    return [
        'form' => $form->createView(),
    ];
}

但是我收到了这个例外:

  

在null

上调用成员函数get()

我的控制器扩展了Symfony\Bundle\FrameworkBundle\Controller\Controller

/**
 * {@inheritdoc}
 */
class DefaultController extends Controller
{
...
}

所以我可以访问容器。

在Symfony的代码中放置一些转储,我看到容器已正确设置:

namespace Symfony\Component\DependencyInjection;

/**
 * ContainerAware trait.
 *
 * @author Fabien Potencier <fabien@symfony.com>
 */
trait ContainerAwareTrait
{
    /**
     * @var ContainerInterface
     */
    protected $container;

    /**
     * Sets the container.
     *
     * @param ContainerInterface|null $container A ContainerInterface instance or null
     */
    public function setContainer(ContainerInterface $container = null)
    {
        dump('Here in the ContainerAwareTrait');
        dump(null === $container);
        $this->container = $container;
    }
}

此转储

Here in the ContainerAwareTrait
false

因此自动装配效果很好并设置容器。

但在ControllerTrait我有这个:

trait ControllerTrait
{
    /**
     * Generates a URL from the given parameters.
     *
     * @param string $route         The name of the route
     * @param mixed  $parameters    An array of parameters
     * @param int    $referenceType The type of reference (one of the constants in UrlGeneratorInterface)
     *
     * @return string The generated URL
     *
     * @see UrlGeneratorInterface
     */
    protected function generateUrl($route, $parameters = array(), $referenceType = UrlGeneratorInterface::ABSOLUTE_PATH)
    {
        dump('Here in the ControllerTrait');
        die(dump(null === $this->container));
        return $this->container->get('router')->generate($route, $parameters, $referenceType);
    }

    ...

这是转储:

Here in the ControllerTrait
true

所以此处containernull,这会导致错误。

任何人都可以帮我解决这个问题吗?

为什么container为空?

如果可能有帮助,这是services.yml配置(Symfony附带的默认配置):

# controllers are imported separately to make sure they're public
# and have a tag that allows actions to type-hint services
AppBundle\Controller\:
    resource: '../../src/AppBundle/Controller'
    public: true
    tags: ['controller.service_arguments']

此问题已在Symfony's issue tracker上发布为问题。

2 个答案:

答案 0 :(得分:0)

S3.3自动装配功能使控制器更容易定义为服务。

将控制器定义为服务的通常动机是避免注入容器。换句话说,您应该明确地注入控制器使用的每个服务。 autowire功能允许您使用操作方法注入,因此您不必在构造函数中注入一堆东西。

但是,基本的Symfony控制器类提供了许多辅助函数,它们使用了大约12种不同的服务。确实一次注射这些是痛苦的。我有点想过,自动装载能力可以为你解决这个问题,但我猜不是。

因此,您基本上需要在服务定义中添加对setContainer的调用。类似的东西:

AppBundle\Controller\:
    resource: '../../src/AppBundle/Controller'
    public: true
    [[setContainer, ['@service_container']]]
    tags: ['controller.service_arguments']

autowire功能正在进行中,所以如果3.4 / 4.0的变化我不会感到惊讶。

答案 1 :(得分:0)

此问题由PR #23239修正,并在Symfony 3.3.3中重新发布。