本文旨在指导读者如何在 symfony 5 应用程序中灵活实现邮件的同步与异步发送。通过利用 symfony messenger 组件,创建自定义消息类和对应的处理器,并进行精确的路由配置,可以有效分离异步邮件逻辑,同时确保常规邮件发送的同步性,避免重复发送问题,从而优化用户体验和系统性能。
在现代 Web 应用中,邮件发送通常是耗时操作,将其异步化可以显著提升用户体验和应用响应速度。Symfony 5 及其更高版本通过集成 Mailer 和 Messenger 组件,提供了强大的邮件发送能力,并支持灵活的同步与异步处理。本文将详细介绍如何配置和实现这一机制,确保您能够根据业务需求选择合适的邮件发送模式。
Symfony Mailer 组件在发送邮件时,默认会生成一个 Symfony\Component\Mailer\Messenger\SendEmailMessage 消息,并将其分发到 Messenger 总线。如果您的 Messenger 配置中将此消息路由到异步传输,那么所有通过 MailerInterface::send() 方法发送的邮件都将被异步处理。这在某些场景下是期望行为,但在需要同时支持同步和异步邮件发送时,则需要更精细的控制。
例如,如果您的 messenger.yaml 配置如下:
# config/packages/messenger.yaml
framework:
messenger:
transports:
async: '%env(MESSENGER_TRANSPORT_DSN)%'
routing:
'Symfony\Component\Mailer\Messenger\SendEmailMessage': async此时,所有邮件都将通过 async 传输进行异步发送。为了实现选择性异步,我们需要引入自定义消息。
为了将邮件发送的具体数据封装起来,并使其能够在 Messenger 总线中传递,我们需要定义一个自定义的消息类。这个类将包含构建 TemplatedEmail 所需的所有信息。
// src/Message/EmailAsync.php
subject = $subject;
$this->bodyHtmlTemplate = $bodyHtmlTemplate;
$this->bodyTextTemplate = $bodyTextTemplate;
$this->recipient = $recipient;
$this->context = $context;
$this->sender = $sender;
}
// 为所有属性生成公共的 getter 方法
public function getSubject(): string
{
return $this->subject;
}
public function getBodyHtmlTemplate(): string
{
return $this->bodyHtmlTemplate;
}
public function getBodyTextTemplate(): string
{
return $this->bodyTextTemplate;
}
public function getRecipient(): string
{
return $this->recipient;
}
public function getContext(): array
{
return $this->context;
}
public function getSender(): ?string
{
return $this->sender;
}
}消息处理器负责接收特定类型的消息,并执行相应的业务逻辑。对于 EmailAsync 消息,其处理器将使用 MailerInterface 来实际构建并发送邮件。
// src/MessageHandler/EmailAsyncHandler.php
mailer = $mailer;
$this->defaultSenderEmail = $defaultSenderEmail;
}
public function __invoke(EmailAsync $emailAsync): void
{
$sender = $emailAsync->getSender() ?? $this->defaultSenderEmail;
$emailToSend = (new TemplatedEmail())
->from(new Address($sender))
->to(new Address($emailAsync->getRecipient()))
->subject($emailAsync->getSubject())
->htmlTemplate($emailAsync->getBodyHtmlTemplate())
->textTemplate($emailAsync->getBodyTextTemplate())
->context($emailAsync->getContext());
$this->mailer->send($emailToSend);
}
}注意事项:
现在,我们需要告诉 Symfony Messenger,当它遇到 App\Message\EmailAsync 类型的消息时,应该将其路由到 async 传输。同时,非常重要的一点是,不要将 Symfony\Component\Mailer\Messenger\SendEmailMessage 路由到异步传输,以确保通过 MailerInterface::send() 直接发送的邮件保持同步。
# config/packages/messenger.yaml
framework:
messenger:
transports:
# 定义异步传输,通常使用 DSN 从环境变量获取连接字符串
async: '%env(MESSENGER_TRANSPORT_DSN)%'
routing:
# 将自定义的 EmailAsync 消息路由到异步传输
'App\Message\EmailAsync': async
# 保持 Symfony\Component\Mailer\Messenger\SendEmailMessage 不被路由到异步
# 这样,通过 MailerInterface::send() 直接发送的邮件将默认同步处理
# 如果不在这里明确指定,它会走默认的同步处理流程。
# 如果需要对所有邮件做统一处理,可以考虑在这里添加一个 'sync' 传输,
# 或者干脆不配置它,让其走默认的同步处理。
# 'Symfony\Component\Mailer\Messenger\SendEmailMessage': sync # 示例:明确路由到同步传输通过上述配置,只有 EmailAsync 消息才会被推入异步队列。而任何直接调用 MailerInterface::send() 发送的 TemplatedEmail,由于其对应的 SendEmailMessage 未被路由到异步传输,将会在当前请求中同步发送。
现在,您可以在应用程序的不同服务中根据需要选择同步或异步发送邮件。
在需要异步发送邮件的服务中,您应该注入 MessageBusInterface,并分发 EmailAsync 消息。
// src/Service/MailManagerAsync.php
bus = $bus;
$this->defaultSenderEmail = $defaultSenderEmail;
}
/**
* 异步发送邮件
*
* @param string $subject 邮件主题
* @param string $bodyHtmlTemplate HTML 模板路径
* @param string $bodyTextTemplate 纯文本模板路径
* @param string $to 收件人邮箱
* @param array $context 传递给模板的上下文数据
* @param string|null $sender 可选,自定义发件人邮箱
*/
public function sendAsyncEmail(
string $subject,
string $bodyHtmlTemplate,
string $bodyTextTemplate,
string $to,
array $context = [],
?string $sender = null
): void {
$emailAsync = new EmailAsync(
$subject,
$bodyHtmlTemplate,
$bodyTextTemplate,
$to,
$context,
$sender ?? $this->defaultSenderEmail
);
$this->bus->dispatch($emailAsync);
}
}对于需要同步发送的邮件,您只需注入 MailerInterface 并直接调用其 send() 方法即可。
// src/Service/MailManagerSync.php
mailer = $mailer;
$this->defaultSenderEmail = $defaultSenderEmail;
}
/**
* 同步发送邮件
*
* @param string $subject 邮件主题
* @param string $bodyHtmlTemplate HTML 模板路径
* @param string $bodyTextTemplate 纯文本模板路径
* @param string $to 收件人邮箱
* @param array $context 传递给模板的上下文数据
* @param string|null $sender 可选,自定义发件人邮箱
*/
public function sendSyncEmail(
string $subject,
string $bodyHtmlTemplate,
string $bodyTextTemplate,
string $to,
array $context = [],
?string $sender = null
): void {
$email = (new TemplatedEmail())
->from(new Address($sender ?? $this->defaultSenderEmail))
->to(new Address($to))
->subjec
t($subject)
->htmlTemplate($bodyHtmlTemplate)
->textTemplate($bodyTextTemplate)
->context($context);
$this->mailer->send($email);
}
}通过上述方法,您已经成功在 Symfony 应用程序中实现了邮件的同步与异步发送分离:
重要提示:
php bin/console messenger:consume async --time-limit=3600
遵循这些步骤和最佳实践,您将能够在 Symfony 应用中高效且灵活地管理邮件发送,从而提升整体性能和可靠性。
# php
# redis
# html
# 处理器
# 编码
# app
# 工具
# ai
# 路由
# 环境变量
# 邮箱
# red
# symfony
# rabbitmq
# 封装
# 数据封装
# 并发
# 异步
# 自定义
# 邮件发送
# 发送邮件
# 您的
# 应用程序
# 可选
# 将其
# 更高
# 这是
# 直接发送
相关文章:
制作国外网站的软件,国外有哪些比较优质的网站推荐?
魔毅自助建站系统:模板定制与SEO优化一键生成指南
如何通过网站建站时间优化SEO与用户体验?
建站之星如何实现PC+手机+微信网站五合一建站?
javascript中对象的定义、使用以及对象和原型链操作小结
小捣蛋自助建站系统:数据分析与安全设置双核驱动网站优化
湖北网站制作公司有哪些,湖北清能集团官网?
网站按钮制作软件,如何实现网页中按钮的自动点击?
建站之星ASP如何实现CMS高效搭建与安全管理?
如何通过主机屋免费建站教程十分钟搭建网站?
如何获取上海专业网站定制建站电话?
高防服务器租用指南:配置选择与快速部署攻略
制作网站的软件下载免费,今日头条开宝箱老是需要下载怎么回事?
韩国服务器如何优化跨境访问实现高效连接?
高防网站服务器:DDoS防御与BGP线路的AI智能防护方案
智能起名网站制作软件有哪些,制作logo的软件?
如何在万网ECS上快速搭建专属网站?
如何高效完成自助建站业务培训?
如何在Golang中处理模块冲突_解决依赖版本不兼容问题
C++如何编写函数模板?(泛型编程入门)
高端建站三要素:定制模板、企业官网与响应式设计优化
长春网站建设制作公司,长春的网络公司怎么样主要是能做网站的?
制作网站公司那家好,网络公司是做什么的?
红河网站制作公司,红河事业单位身份证如何上传?
网站制作模板下载什么软件,ppt模板免费下载网站?
b2c电商网站制作流程,b2c水平综合的电商平台?
免费网站制作模板下载,除了易企秀之外还有什么H5平台可以制作H5长页面,最好是免费的?
香港服务器WordPress建站指南:SEO优化与高效部署策略
怎么制作网站设计模板图片,有电商商品详情页面的免费模板素材网站推荐吗?
已有域名和空间,如何快速搭建网站?
建站之星后台密码遗忘如何找回?
黑客入侵网站服务器的常见手法有哪些?
西安专业网站制作公司有哪些,陕西省建行官方网站?
如何通过FTP空间快速搭建安全高效网站?
如何在万网开始建站?分步指南解析
电视网站制作tvbox接口,云海电视怎样自定义添加电视源?
成都网站制作公司哪家好,四川省职工服务网是做什么用?
完全自定义免费建站平台:主题模板在线生成一站式服务
如何快速搭建虚拟主机网站?新手必看指南
香港服务器网站卡顿?如何解决网络延迟与负载问题?
如何将凡科建站内容保存为本地文件?
购物网站制作费用多少,开办网上购物网站,需要办理哪些手续?
韩国代理服务器如何选?解析IP设置技巧与跨境访问优化指南
如何做网站制作流程,*游戏网站怎么搭建?
如何在服务器上配置二级域名建站?
网站制作免费,什么网站能看正片电影?
建站之星价格显示格式升级,你的预算足够吗?
西安大型网站制作公司,西安招聘网站最好的是哪个?
如何快速查询网站的真实建站时间?
如何在橙子建站上传落地页?操作指南详解
*请认真填写需求信息,我们会在24小时内与您取得联系。