问候,我正在开发一个网络应用程序。其中一部分将允许用户安排在一天中的特定时间发送给他们的“提醒”电子邮件。完成此任务的最佳方法是什么?基本上,当我想要的是“中断”模式时,我提出的所有解决方案都以“轮询”模式运行。
以下是我提出的一些可能的解决方案:
每分钟都有一个cronjob火。触发的脚本检查数据库以查看是否有要发送的电子邮件,如果有,则发送它们,否则它会重新进入休眠状态。这样做的缺点是每分钟都会产生一些开销。此外,这可能不是一个可扩展的系统,尤其是当用户数量过大以至于发送所有电子邮件可能需要一分多钟时。
与#1相同,但作业仅每15分钟触发一次。这有点易于管理,但并不完美,因为它限制了用户在15分钟标记上的提醒,并且在没有要发送的电子邮件时仍然会产生一些开销。不错,但也不完美。
让PHP exec()有一些动态改变crontab或在底层linux中调度“at”作业的代码。这将给我灵活性和“中断”类型模型我非常渴望,但会在允许PHP执行exec()linux代码方面打开一个巨大的安全漏洞。所以,我要继续把这个问题排除在外。
那么,还有什么比我想出的更好?也许是一种不使用cron来安排电子邮件的方法?我非常想知道你们对此有什么看法:)。
答案 0 :(得分:3)
您可以拥有一个仍在运行的PHP脚本。每隔设置的时间间隔,查询数据库中是否有需要在下一个时间间隔内发送的电子邮件。将其分解为每分钟一组的数组。因此,如果您选择15分钟,您将拥有一个包含15个条目的数组,每个条目都包含当时需要发送的所有电子邮件。
然后,您可以使用forking来分割流程,一个处理发送电子邮件,另一个处理直到下一分钟,然后再次拆分。为了扩展,你可以分叉多个进程,每个进程处理一定数量的电子邮件。
简而言之,一个进程管理队列并分叉其他进程来处理发送。当队列“空”时,它会变得更多。你可以定期运行一个cron来确保进程没有死亡。
答案 1 :(得分:2)
使用第一个变体。
发送可能需要一分多钟 所有的电子邮件
不考虑开销 - 不是在这种情况下。
答案 2 :(得分:2)
在选项#1和#2中使用cron没有什么特别的错误,我不知道你正在使用什么类型的应用程序,但是让用户能够安排到确切的分钟可能没有必要。即便如此,如果您的脚本将提醒状态标记为“待处理”等,并且脚本的任何新实例仅发送非“待处理”或“已发送”的实例,则可能不会出现问题。
您可以使用Hudson或类似的应用程序来帮助您进行脚本管理,并使您能够密切关注故障等。它甚至可以在出现故障时发出通知。它支持它自己的基于java的cron系统。
如果应用确实变大,您当然可能希望将此过程卸载到与Web服务器不同的服务器上。如果您尚未使用外部SMTP服务,也可能需要查看用于发送邮件的第三方工具,并查看它们可能具有的集成工具。这也应该提高交付率等。
答案 3 :(得分:1)
请勿混淆排队邮件,以便发送邮件并实际发送邮件。
您的邮件服务器可能需要fifteen minutes才能发送一封电子邮件。但是我的mail(1)
仅0.036s
需要对邮件进行排队以便发送。
即使你每分钟发送超过1600封电子邮件(干得好!),你可以稍微调整你的代码,以便在几分钟前开始发送提醒电子邮件,预期会出现“尖峰” - 比如说,在你的数据库中向前看五分钟,看看是否有> 1000封邮件,并以1/5概率,1/4概率,1/3概率,1/2概率开始排队,然后排队余。
答案 4 :(得分:0)
有一个命令hostman
,允许您在特定时间调用函数。那应该做你想要的。
这是我最初的建议:
组合怎么样?