- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我正在从事 Symfony 2 WebApp 项目。集成商店的逻辑在 MyShopBundle
中实现。
在将 Twig Extension
添加到 bundle 后,我得到一个异常:
我完全理解这条消息的意思,但到目前为止我没有找到解决问题的方法:
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 中的循环引用。但它们都与使用一些常见的 Bundle/Services 的一些特殊情况有关,例如 Doctrine
、FOSUserBundle
等。这些线程中讨论的解决方案在这里不起作用。
显而易见的解决方案是将 PaymentService
分成两部分:一部分包含 isPaymentComplete(int userId)
另一部分提供 updatePaymentStatus
。但这并不像听起来那么容易,因为这两种方法使用其他常见的 PaymentService
方法。为了避免将杠杆上的循环引用更深地移动,我需要将 PaymentService
分成三个部分:
isPaymentComplete
updatePaymentStatus
这(很有可能)会奏效,但会非常、非常难看。这三个服务中的所有方法都服务于相同的目的(处理付款),因此应该包含在相同的服务中。将服务分解成不同的部分与代码结构无关,而只会通过破解来分解循环引用。
我已经尝试将引用从构造函数移至 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 }
结果还是一样。
那么:是否有任何干净的解决方案来解决这个引用问题?
最佳答案
这个问题意味着您将需要更多的服务和更少的依赖。例如,您可以拥有像 clientMailProvider 这样的服务,它会获取 twig html 电子邮件并将其与您的数据混合。例如,该服务将由商店支付服务或其他服务调用。
现在针对你的问题,我认为你可以注入(inject) 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));
...
}
关于交响乐 2 : Resolve circular reference,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/36977046/
我使用 resolver() 作为 socket() 的替代方法,因为我发现当多个连接建立到不同的 IP 时,它最终会停止工作。 无论如何它会向我返回一个警告,我应该使用 dns.resolver.R
我有这个代码: var promise1 = new Promise(function(resolve, reject) { setTimeout(() => { console
我仍在学习 PHP,我认为我不是母语人士,这并不难理解。 此时,看了一大堆文档,跳入了深水区,于是打开Laravel源文件,一个接一个地看,试图更好地理解MVC的整个实现,包括路由、中间件如何组合成一
public JsonResult GetEvents(double start, double end) { var userName = Session["UserName"] as st
我正在使用 bluebird,我看到了两种将同步函数解析为 Promise 的方法,但我不明白这两种方法之间的区别。看起来堆栈跟踪有点不同,所以它们不仅仅是一个别名,对吧? 那么首选的方式是什么? 方
我写了下面的代码: function readFile(path) { return new Promise(function(resolve, reject){ if(!fs
我正在使用 bluebird,我看到了两种将同步函数解析为 Promise 的方法,但我不明白这两种方法之间的区别。看起来堆栈跟踪有点不同,所以它们不仅仅是一个别名,对吧? 那么首选的方式是什么? 方
在某处读过这个例子: return new Promise( (resolve, reject) => { fs.readFile(file, (err, data) => { if (e
我刚开始学习 React,我一直在尝试让我的 React 应用程序连接到我的数据库 var mysql = require('mysql'); var con = mysql.createConnec
我需要从 $http 调用中返回一个 promise 中的自定义响应,以便我可以链接更多调用。我有两个可用的实现。有人可以解释两者之间的区别,并争论其中一个更好吗? 在 fooService.js 实
免责声明:这里实际上提出了两个问题,但我觉得它们密切相关。 我正在尝试将 promise 对象传递给指令,并且我想在 promise 解析后立即在指令中运行一些初始化代码。 在我的 Controlle
我正在尝试创建类似于 this code 的东西在 boost.asio 示例中找到。 套接字.h: class some_class { private: ... boost
正如我们所知,Promise 构造函数采用一个执行函数,该函数具有两个参数,我们使用它们来生成成功案例或失败案例。今天我在编程,我被卡住了,但后来我解决了这个问题,但我发现了一件事需要理解。 有什么区
我认为 Promise.resolve 和 new Promise(resolve) 可以互换。 考虑一下: A. new RSVP.Promise(function (resolve, reject
我下载了一个 Java 项目,我想研究并从中学习一些东西。当我在另一台计算机上下载它时效果很好,但是当我在我的计算机上尝试时,几乎每个声明和导入都会给出错误消息“* cannot be resolve
我昨天看到了一些有趣的编译器行为,我想我明白为什么会这样,但我想确定一下。所以,我不会写我的推理,只写事实。 请注意,我使用 vector 而不是 string 并不是错字。我是故意这样做的,这样编译
我正在尝试运行 Ember 测试,它给出了这个错误,提示无法找到从 `AppName/resolver 导入的模块 ember-resolver。 我不确定是什么原因造成的。我正在使用 Ember-c
Code#1 和 Code#2 的区别在于:Code#1 使用 resolve(p) 而 Code#2 使用 p.then(()=>resolve()) 。我希望输出序列是不变的,但它们会生成不同的序
IntelliJ IDEA 无法解析内置 JVM 类型和方法的常见原因有哪些?例如,当我将鼠标悬停在 String 上时,工具提示显示“无法解析符号“String””。就好像 IntelliJ 不知道
IntelliJ IDEA 无法解析内置 JVM 类型和方法的常见原因有哪些?例如,当我将鼠标悬停在 String 上时,工具提示显示“无法解析符号“String””。就好像 IntelliJ 不知道
我是一名优秀的程序员,十分优秀!