37

我搜索并找不到答案。我的应用程序中有数据库角色模型。用户可以有一个角色,但这个角色必须存储到数据库中。

但是随后用户需要从数据库中添加默认角色。所以我创建了一个服务:

<?php

namespace Alef\UserBundle\Service;

use Alef\UserBundle\Entity\Role;

/**
 * Description of RoleService
 *
 * @author oracle
 */
class RoleService {

    const ENTITY_NAME = 'AlefUserBundle:Role';

    private $em;

    public function __construct(EntityManager $em)
    {
        $this->em = $em;
    }

    public function findAll()
    {
        return $this->em->getRepository(self::ENTITY_NAME)->findAll();
    }

    public function create(User $user)
    {
        // possibly validation here

        $this->em->persist($user);
        $this->em->flush($user);
    }

    public function addRole($name, $role) {
        if (($newrole = findRoleByRole($role)) != null)
            return $newrole;
        if (($newrole = findRoleByName($name)) != null)
            return $newrole;

        //there is no existing role
        $newrole = new Role();
        $newrole->setName($name);
        $newrole->setRole($role);

        $em->persist($newrole);
        $em->flush();

        return $newrole;
    }

    public function getRoleByName($name) {
        return $this->em->getRepository(self::ENTITY_NAME)->findBy(array('name' => $name));
    }

    public function getRoleByRole($role) {
        return $this->em->getRepository(self::ENTITY_NAME)->findBy(array('role' => $role));
    }

}

services.yml的是:

alef.role_service:
    class: Alef\UserBundle\Service\RoleService
    arguments: [%doctrine.orm.entity_manager%]

现在我想在两个地方使用它: UserControllerUser实体。我怎样才能让它们进入实体?至于控制器,我想我只需要:

$this->get('alef.role_service');

但是如何在实体内部获得服务?

4

3 回答 3

46

你没有。这是一个很常见的问题。实体应该只知道其他实体,而不是实体管理器或其他高级服务。过渡到这种开发方式可能有点挑战,但通常是值得的。

您要做的是在加载用户时加载角色。通常,您最终会得到一个执行此类操作的 UserProvider。您是否阅读过有关安全的部分?这应该是你的起点:

http://symfony.com/doc/current/book/security.html

于 2012-04-26T12:25:23.687 回答
37

首先将服务引入实体如此困难的原因是 Symfony 的明确设计意图是永远不应该在实体内部使用服务。因此,最佳实践答案是重新设计您的应用程序,使其不需要在实体中使用服务。

但是,我发现有一种方法可以做到这一点,而不涉及弄乱全局内核。

Doctrine 实体具有生命周期事件,您可以将事件侦听器挂钩,请 参阅http://docs.doctrine-project.org/projects/doctrine-orm/en/latest/reference/events.html#lifecycle-events在示例中,我将使用 postLoad,它会在创建实体后不久触发。

EventListeners可以作为你注入其他服务的服务。

添加到 app/config/config.yml:

services:
     example.listener:
           class: Alef\UserBundle\EventListener\ExampleListener
     arguments:
           - '@alef.role_service'
     tags:
           - { name: doctrine.event_listener, event: postLoad }

添加到您的实体:

 use Alef\UserBundle\Service\RoleService;

 private $roleService;

 public function setRoleService(RoleService $roleService) {
      $this->roleService = $roleService;
 }

并添加新的 EventListener:

namespace Alef\UserBundle\EventListener;

use Doctrine\ORM\Event\LifecycleEventArgs;
use Alef\UserBundle\Service\RoleService;

class ExampleListener
{
     private $roleService;

     public function __construct(RoleService $roleService) {
         $this->roleService = $roleService;
     }

     public function postLoad(LifecycleEventArgs $args)
     {
         $entity = $args->getEntity();
         if(method_exists($entity, 'setRoleService')) {
             $entity->setRoleService($this->roleService);
         }
     }
}

请记住,此解决方案附带一个警告,即这仍然是一种快速而肮脏的方式,实际上您应该考虑以正确的方式重新设计您的应用程序。

于 2015-09-29T18:54:10.220 回答
1

感谢Kai 在上面对问题的回答,但它与 symfony 5.x 不兼容。

准确地说这是一个不好的做法,但在某些特殊情况下需要,比如遗留代码或糟糕的数据库设计(作为模式迁移之前的临时解决方案)

就我而言,我将此代码与邮件程序和翻译器一起使用,如果 Symfony >= 5.3 会引入私有属性的问题,因此这里是 symfony 最新版本的解决方案:

在 config/services.yaml 中:

services:
    Alef\UserBundle\EventListener\ExampleListener:
        tags:
           - { name: doctrine.event_listener, event: postLoad }

示例监听器:

namespace Alef\UserBundle\EventListener;

use Doctrine\ORM\Event\LifecycleEventArgs;
use Alef\UserBundle\Entity\Role;

class ExampleListener
{

    public function postLoad(LifecycleEventArgs $postLoad): void
    {
        $entity = $postLoad->getEntity();
        if ($entity instanceof User) {
            $repository = ;
            $entity->roleRepository(
                $postLoad->getEntityManager()->getRepository(Role::class)
            );
        }
    }
}

在您的实体中(或者如果您在多个实体中使用它,则在特征中):


    use Alef\UserBundle\Service\RoleService;

    /** @internal bridge for legacy schema */
    public function roleRepository(?RoleRepository $repository = null) {
        static $roleRepository;
        if (null !== $repository) {
            $roleRepository = $repository;
        }
        return $roleRepository;
    }

    public function getRoleByName($name) {
        return $this->roleRepository()->findBy(array('name' => $name));
    }
于 2021-08-02T14:36:19.327 回答