- android - RelativeLayout 背景可绘制重叠内容
- android - 如何链接 cpufeatures lib 以获取 native android 库?
- java - OnItemClickListener 不起作用,但 OnLongItemClickListener 在自定义 ListView 中起作用
- java - Android 文件转字符串
我有一个网络应用程序 (PHP/MYSQL),每次有人通过我的应用程序中的表单在我的数据库中提交条目时,它都会向与该条目关联且选择接收通知的每个人发送一封通知电子邮件。
例如,有一个可以容纳 15 人的橄榄球采摘池。当第 16 个人提交他们的参赛作品时,0-15 人之间的任何地方都会收到有新参赛作品的通知。可能是 1 个,可能是全部 15 个,可能是其中的一半左右。
但我的站点可能有 100 个足球池,一些有 5 人,一些有 100 人。
所以我设置了一个单独的表来记录每个条目的基本信息,并将“已发送”列设置为“0”。
然后我让下面的 cronjob 每分钟运行一次,它会查找所有具有“0”的条目(这意味着与该条目关联的人尚未收到通知)并循环并向每个条目发送一封唯一的电子邮件关联的人。我发送独特的电子邮件,因为每封电子邮件都有一个退订链接和特定于该人的信息。发送电子邮件后,它将所有这些条目的“已发送”列更新为“1”,因此它将被忽略。
我通过 Amazon SES 使用 PHPMailer SMTP 发送电子邮件。
<?php
require_once("includes/session.php");
require_once("includes/connection.php");
require_once("includes/functions.php");
require 'phpmailer/PHPMailerAutoload.php';
//find all new entries with sent = 0, loop thru and send emails
$stmt = $pdo->prepare("SELECT * FROM cron_email_notify WHERE sent = 0");
$stmt->execute();
foreach ($stmt->fetchAll(PDO::FETCH_ASSOC) as $row) {
//bunch of variables set here (which i'll leave out for brevity's sake) to be used in the emails (like firstname, etc)
//Next, send a unique email to all people who have chosen to be notified
$stmt = $pdo->prepare("SELECT * FROM entries WHERE poolid = ? AND notify = 'yes'");
$stmt->execute([$poolid]);
foreach ($stmt->fetchAll(PDO::FETCH_ASSOC) as $row) {
//get emails into array
$notifyemailsarray[$row['email']]=$row;
}
/// Send notification email to all people who wish to be notified, if any
$mail = new PHPMailer;
//$mail->SMTPDebug = 3; // Enable verbose debug output
$mail->isSMTP(); // Set mailer to use SMTP
$mail->SMTPKeepAlive = true; //Helps with speed for multiple emails
$mail->Host = 'tls://email-smtp.us-east-1.amazonaws.com'; // Specify main and backup SMTP servers
$mail->SMTPAuth = true; // Enable SMTP authentication
// and so on with other phpmailer info which I cut out for brevity's sake
if (!empty($notifyemailsarray)) {
foreach($notifyemailsarray as $email => $details) {
// Assemble the fullname here
$fullname = $details['firstname'] . ' ' . $details['lastname'];
$mail->addAddress($details['email'], $fullname);
$mail->Body = "(leaving out for brevity's sake)";
$mail->AltBody = "(leaving out for brevity's sake)";
if(!$mail->send()) {
echo 'Message could not be sent.';
echo 'Mailer Error: ' . $mail->ErrorInfo;
} else {
//email sent
}
// Clear all addresses for next loop
$mail->clearAddresses();
}
} else {
//no people are set to be notified
}
$mail->SmtpClose();
//update cron_email_notify table's sent field to 1, so we know not to send again
$stmt = $pdo->prepare("UPDATE cron_email_notify SET sent = 1 WHERE poolid = ? and sent = 0");
$stmt->execute([$poolid]);
}
?>
在测试中一切正常,但我只测试了它发送大约 10 封电子邮件的地方(即小土 bean )
我的应用程序将收到大量提交,并且随着时间的推移应该会不断增长,所以我只是想看看这是否是一个潜在的问题(我对在后台运行 cronjobs 的经验很少,所以不是确定要影响网站性能的操作/流量有多疯狂)。
我很担心,因为发送 10-15 封电子邮件时,上面的 cronjob 脚本似乎需要大约 5 秒才能完成。
现在,假设有大量条目涌入后,我的实时网站启动了一个 cronjob,它必须发送 200 封电子邮件。如果 cronjob 需要一分钟的时间才能运行,下一个 cronjob 将开始(因为它们每分钟运行一次),这难道不会让我的系统完全陷入困境吗?或者同一个 cronjob 可以在另一个 cronjob 运行时没有问题吗?
显然,许多网站都会发送大量通知电子邮件,所以这一定是一个普遍的需求,我确信有一个很好的解决方案,所以只是好奇任何人的想法/建议/等等。也许我这样做的方式很好,直到我获得如此多的流量,这将是一个很大的问题。或者也许我现在可以做更多的事情来帮助抵消可能陷入困境的系统?
最佳答案
我在一家公司工作,该公司为美国一半以上的公立学校发送通知。我们每天发送大约 1000 万封电子邮件。
您的代码是单线程的,因此它会连续发送电子邮件,一封接一封,这可能需要一段时间。你问如果它花费的时间比你的 cron 作业之间的间隔时间长会发生什么?也就是说,如果您的 cron 作业每分钟运行一次,但在某个时候它会在前一个脚本仍在运行时启动您的脚本。然后,您的脚本的两次调用都将处理同一批电子邮件,您的用户将开始收到重复的电子邮件。
因此,最好运行一个守护进程。这只是意味着不是 cron 每分钟启动 PHP 脚本,而是编写 PHP 脚本以运行 while (true)
循环并且永不退出。在循环结束时,使脚本 sleep()
最多 60 秒。然后它转到循环的顶部,检查数据库以查看是否有要运行的电子邮件批处理,然后处理该批处理。然后它重新进入休眠状态,持续的秒数等于 60 减去处理批处理所花费的时间(但不少于零秒)。
这样一来,如果出现激增,即使一批电子邮件需要 75 秒,它也永远不会同时运行两个应用程序。您的守护进程的 sleep 计算将使它至少 sleep 0 秒,然后它会立即开始下一批。希望激增是暂时的,应用程序可以在某个时候 catch 。否则,您的应用将越来越落后于计划。
您的应用应将发送的电子邮件数量以及批处理之间休眠的时间写入日志文件。您应该监控此文件,以便在它无法满足需求时知道。一旦您变得真正专业,您就会将这些数据放入 Grafana 仪表板或类似的东西中。
如果您的流量增长到单线程应用始终无法在 60 秒内发送一批电子邮件的程度,那么您将不得不并行运行多个电子邮件发件人应用。 PHP 在多线程编程方面表现不佳,因此您可能希望使用 Java 或 Go。
尝试让多个线程轮询数据库是个坏主意,因为您会遇到竞争条件和锁争用。我们在我公司所做的是让一个线程轮询数据库。当它找到一批要发送的电子邮件时,它会将它们推送到消息队列中(我们使用 ActiveMQ,但还有其他的)。消息队列使多个线程可以轻松地从队列中拉出项目。然后,多个并发电子邮件发送应用程序线程就可以轻松地并行处理成批电子邮件。
P.S.:您是否意识到,即使部分或全部电子邮件发送失败,您的代码也会将池标记为 sent=1
?
关于php - cronjob 是使用 SMTP 通过 PHPMailer 发送大量通知电子邮件的正确方法吗?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47584277/
大家好,我是 php 和 html 新手,面临问题:SMTP 错误:无法连接到 SMTP 主机。邮件程序错误:SMTP 错误:无法连接到 SMTP 主机。但在本地 xampp 中工作正常,在服务器中出
我希望这个问题不是完全偏离主题。 我一直想知道 SMTP 电子邮件验证。有很多服务(例如 Kickbox.io、Email-Validator.net 等)似乎可以处理 SMTP 验证,而不会阻止其
我正在尝试使用 phpMailer 通过电子邮件向用户发送确认消息。我的代码是这样的: IsSMTP(); // set mailer to use SMTP $mail->Host = "ssl:/
我已经在我的本地机器(Windows 7)上安装了 bugzilla 并且它运行良好。但是当我尝试创建一个新帐户时,它说 There was an error sending mail from it
我有一个将发送大量输出的程序。我只是想知道最大电子邮件附件大小是多少?根据RFC 1870邮件服务器可以拒绝邮件,因为它们太大了,但是使用 SMTP/MIME 时有最大大小吗?我在这方面找不到任何东西
假设我有一个网站,其某些功能需要电子邮件验证(例如用户注册)。当然,我会使用正则表达式验证电子邮件,但曾几何时,我在其他人的代码中看到了 SMTP 验证。 SMTP 验证有哪些优点和缺点? 我可以假设
关闭。这个问题不满足Stack Overflow guidelines .它目前不接受答案。 想改善这个问题吗?更新问题,使其成为 on-topic对于堆栈溢出。 2年前关闭。 Improve thi
已关闭。此问题不符合Stack Overflow guidelines 。目前不接受答案。 这个问题似乎不是关于 a specific programming problem, a software
我正在尝试使用 Vala 和 GLib + GIO 实现一个简单的 SMTP 服务器。 到目前为止,纯文本通信没有问题,但当涉及使用 STARTTLS 的 TLS 时,事情会变得更加困难。 这是我到目
我使用 go 的 net/smtp 发送电子邮件,它对某些电子邮件工作正常但对其他电子邮件无效。我收到 554 5.5.1 Error: no valid recipients 但我很确定我提供了正确
我需要帮助 这是我的代码: require 'PHPMailerAutoload.php'; $mail = new PHPMailer; $mail->isSMTP(); $mail->Host =
我的 joomla 联系表 gmail SMTP 设置不起作用。请参阅下面提交联系表时显示的消息。应该是什么原因?extension=php_openssl.dll 也启用了.. 错误信息-: SMT
我正在尝试使用 Amazon 的 SES/SMTP 发送电子邮件,但出现以下错误: javax.mail.MessagingException:无法连接到 SMTP 主机:email-smtp.us-
在我的Mac终端上,我试图通过telnet将smtp.gmail.com转换为port 587。 在Google Apps(设置为管理Dreamhost域)上,我已配置了中继,如下所示: "Allow
理论上,Request For Comments (RFC) 集包含开发人员构建 SMTP 客户端需要知道的一切。然而,要知道哪些 RFC 需要考虑,哪些可以忽略并不总是那么容易。 有没有人有 RFC
MT 支持 SMTP SendMail,还是我坚持使用 MFMailComposeViewController?现在,我可以使用它(MFMailComposeViewController),但是当我添
很难说出这里问的是什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或言辞激烈,无法以目前的形式合理回答。如需帮助澄清此问题以便可以重新打开,visit the help center . 10年前关
我正在尝试通过 BizTalk SMTP 发送端口发送消息。具体来说,我正在通过编排的“稍后指定”端口发送消息。我的目标是使用我选择的附件文件名将消息正文附加到已发送的电子邮件。 但是,无论我尝试什么
我正在编写一个通过有效的 GMail 用户 ID 和密码发送邮件的应用程序。 我只是想在 Windows XP 命令行上模拟 SMTP 连接,当我在 465 端口 telnet smtp.gmail.
关于从应用程序通过 GMails SMTP 服务器发送的电子邮件被标记为垃圾邮件的问题已经有很多讨论。 阅读其他帖子我无法弄清楚我的问题。我的电子邮件最终仍然是 SPF 中性的。 在我的 *(捕获所有
我是一名优秀的程序员,十分优秀!