Symfony 2 EntityManager инъекция в сервисе
Я создал свой собственный сервис, и мне нужно ввести doctrine EntityManager, но я этого не вижу __construct() вызывается на мой сервис, и инъекция не работает.
вот код и конфиги:
<?php
namespace TestCommonBundleServices;
use DoctrineORMEntityManager;
class UserService {
/**
*
* @var EntityManager
*/
protected $em;
public function __constructor(EntityManager $entityManager)
{
var_dump($entityManager);
exit(); // I've never saw it happen, looks like constructor never called
$this->em = $entityManager;
}
public function getUser($userId){
var_dump($this->em ); // outputs null
}
}
здесь services.yml в моей пачке
services:
test.common.userservice:
class: TestCommonBundleServicesUserService
arguments:
entityManager: "@doctrine.orm.entity_manager"
я импортировал это .yml в config.yml в моем приложении, как это
imports:
# a few lines skipped, not relevant here, i think
- { resource: "@TestCommonBundle/Resources/config/services.yml" }
и когда я вызываю службу в контроллере
$userservice = $this->get('test.common.userservice');
$userservice->getUser(123);
Я получаю объект (не null), но $this->em in UserService-это null, и, как я уже упоминал, конструктор на UserService никогда не вызывался
еще одна вещь, контроллер и UserService находятся в разных пакетах (мне действительно нужно, чтобы проект был организован), но все же: все остальное работает нормально, я даже могу позвонить
$this->get('doctrine.orm.entity_manager')
в том же контроллере, который я использую, чтобы получить UserService и получить действительный (не нулевой) объект EntityManager.
похоже, что мне не хватает части конфигурации или какой-то связи между UserService и Doctrine config.
4 ответов:
метод конструктора вашего класса должен быть вызван
__construct(), а не__constructor():public function __construct(EntityManager $entityManager) { $this->em = $entityManager; }
для современной справки, в Symfony 2.4+, вы не можете назвать аргументы для метода инъекции конструктора больше. Согласно документация вы бы прошли в:
services: test.common.userservice: class: Test\CommonBundle\Services\UserService arguments: [ "@doctrine.orm.entity_manager" ]и тогда они будут доступны в том порядке, в котором они были перечислены через аргументы (если их больше 1).
public function __construct(EntityManager $entityManager) { $this->em = $entityManager; }
примечание по состоянию на Symfony 3.3 EntityManager амортизируется. Вместо этого используйте EntityManagerInterface.
namespace AppBundle\Service; use Doctrine\ORM\EntityManagerInterface; class Someclass { protected $em; public function __construct(EntityManagerInterface $entityManager) { $this->em = $entityManager; } public function somefunction() { $em = $this->em; ... } }
С 2017 года и Symfony 3.3 вы можете зарегистрировать репозиторий как сервис, со всеми своими преимуществами он имеет.
проверьте мой пост как использовать репозиторий с доктриной в качестве сервиса в Symfony для более общего описания.
в вашем конкретном случае исходный код с настройкой будет выглядеть так:
1. Используйте в своих услугах или контроллере
<?php namespace Test\CommonBundle\Services; use Doctrine\ORM\EntityManagerInterface; class UserService { private $userRepository; // use custom repository over direct use of EntityManager // see step 2 public function __constructor(UserRepository $userRepository) { $this->userRepository = $userRepository; } public function getUser($userId) { return $this->userRepository->find($userId); } }2. Создать новый пользовательский репозиторий
<?php namespace Test\CommonBundle\Repository; use Doctrine\ORM\EntityManagerInterface; class UserRepository { private $repository; public function __construct(EntityManagerInterface $entityManager) { $this->repository = $entityManager->getRepository(UserEntity::class); } public function find($userId) { return $this->repository->find($userId); } }3. Регистрация услуг
# app/config/services.yml services: _defaults: autowire: true Test\CommonBundle\: resource: ../../Test/CommonBundle
Comments