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,如 Doctrine
、FOSUserBundle
等。这些线程中讨论的解决方案在这里不起作用。
显而易见的解决方案是将 PaymentService
分成两部分:一部分包含 isPaymentComplete(int userId)
,另一部分提供 [=29=]。但这并不像听起来那么容易,因为这两种方法使用其他常见的 PaymentService
方法。为了避免将杠杆上的循环参考更深地移动,我需要将 PaymentService
分成三个部分:
- 服务 1 包括
isPaymentComplete
- 服务 2 包括
updatePaymentStatus
- Service3 包括 Service1 和 2 使用的所有常用方法
这会起作用(很有可能),但会非常非常难看。这三个服务中的所有方法都服务于相同的目的(处理付款),因此应该包含在相同的服务中。将服务分解成不同的部分与代码结构无关,而只能通过破解来分解循环引用。
我已经尝试将引用从构造函数移动到 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));
...
}
我正在从事 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,如 Doctrine
、FOSUserBundle
等。这些线程中讨论的解决方案在这里不起作用。
显而易见的解决方案是将 PaymentService
分成两部分:一部分包含 isPaymentComplete(int userId)
,另一部分提供 [=29=]。但这并不像听起来那么容易,因为这两种方法使用其他常见的 PaymentService
方法。为了避免将杠杆上的循环参考更深地移动,我需要将 PaymentService
分成三个部分:
- 服务 1 包括
isPaymentComplete
- 服务 2 包括
updatePaymentStatus
- Service3 包括 Service1 和 2 使用的所有常用方法
这会起作用(很有可能),但会非常非常难看。这三个服务中的所有方法都服务于相同的目的(处理付款),因此应该包含在相同的服务中。将服务分解成不同的部分与代码结构无关,而只能通过破解来分解循环引用。
我已经尝试将引用从构造函数移动到 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));
...
}