Symfony 2:解决循环引用

Symfony 2: Resolve circular reference

我正在从事 Symfony 2 WebApp 项目。综合店铺的逻辑在MyShopBundle.

中实现

向捆绑包添加 Twig Extension 后出现异常:

我完全理解这条消息的意思,但到目前为止我没有找到解决问题的办法:

MyShopBundle 中的 PaymentServices 提供与付款流程相关的各种不同服务。除此之外,它还会观察支付状态并自动向用户发送电子邮件通知。另外它提供 isPaymentComplete(int userId).

要从 Twig 模板呈现电子邮件的内容,PaymentServices 需要引用 Twig_Environment。服务从其构造函数获取此引用:

class MyPaymentService {
    protected $twig;

    public function __construct(\Twig_Environment $twig, ...) {
        $this->twig = $twig;
        ...
    }

    public function updatePaymentStatus(int userId) {
        // Update Payment
        ...

        // Send notification to user
        $mailBody = $this->twig->render('MyShopBundle:Emails:statusUpdate.html.twig', array('user_name' => $username));
        ...
    }

    public function isPaymentComplete(int userId) {
        ...
        return true;   
    }
}

另一方面,Twig Extension 需要引用 PaymentService 来创建 isPaymentComplete(int userId) 的 twig 版本:

class ShopExtension extends \Twig_Extension {
    private $paymentService;

    public function __construct(PaymentService $service, ...) {
        $this->paymentService = $service;
        ...
    }

    public function getName() {
        return 'my_shop_bundle_extension';
    }

    public function getFunctions() {
        $functions = array();

        $functions[] = new \Twig_SimpleFunction('msb_IsPaymentComplete', array(
            $this,
            'msb_IsPaymentComplete'
        ));

        return $functions;
    }

    public function msb_IsPaymentComplete($user) {
        if ($this->paymentService)
            return $this->paymentService->isPaymentComplete($user);
        else 
            return false;
    }
} 

这是service.yml

中的服务定义
services:
    shop.payment.service:
        class: MyShopBundle\Service\PaymentService
        arguments: 
          - "@twig"
          - ...

    app_subscription.twig_extension:
        class: MyShopBundle\Twig\ShopExtension
        arguments: 
            - "@shop.payment.service"
        tags:
          - { name: twig.extension }

异常来源明确:

PaymentService --> Twig --> ShopExtension --> PaymentService

问题是:如何解决?

我发现了其他问题,处理 Symfony 中的循环引用。但它们都与一些特殊情况有关,使用一些常见的 Bundles/Services,如 DoctrineFOSUserBundle 等。这些线程中讨论的解决方案在这里不起作用。

显而易见的解决方案是将 PaymentService 分成两部分:一部分包含 isPaymentComplete(int userId),另一部分提供 [​​=29=]。但这并不像听起来那么容易,因为这两种方法使用其他常见的 PaymentService 方法。为了避免将杠杆上的循环参考更深地移动,我需要将 PaymentService 分成三个部分:

这会起作用(很有可能),但会非常非常难看。这三个服务中的所有方法都服务于相同的目的(处理付款),因此应该包含在相同的服务中。将服务分解成不同的部分与代码结构无关,而只能通过破解来分解循环引用。

我已经尝试将引用从构造函数移动到 setter:

services:
    shop.payment.service:
        class: MyShopBundle\Service\PaymentService
        calls:
          - [ setTwig, [ "@twig" ] ]

    app_subscription.twig_extension:
        class: MyShopBundle\Twig\ShopExtension
        calls:
          - [ setService, [ "@shop.payment.service" ] ]
        tags:
          - { name: twig.extension }

结果还是一样。

所以:是否有任何 clean 解决方案来解决这个引用问题?

这个问题意味着您需要更多的服务和更少的依赖。例如,您可以拥有像 clientMailProvider 这样的服务,它会获取 twig html 电子邮件并将其与您的数据混合。例如,此服务将由商店支付服务或其他服务调用。

现在针对你的问题,我认为你可以注入 twig 的模板部分而不是整个服务:

services:
    shop.payment.service:
        class: MyShopBundle\Service\PaymentService
        arguments: 
          - "@templating"

服务

use Symfony\Component\Templating\EngineInterface;

class MyPaymentService {
    protected $templating;

    public function __construct(EngineInterface $templating, ...) {
        $this->templating= $templating;
        ...
    }

    public function updatePaymentStatus(int userId) {
        // Update Payment
        ...

        // Send notification to user
        $mailBody = $this->templating->render('MyShopBundle:Emails:statusUpdate.html.twig', array('user_name' => $username));
        ...
    }