我知道有很多 API 可以做到这一点,但我也知道托管环境(即 ASP.NET)限制了您可以在单独的线程中可靠地执行的操作。
我可能完全错了,所以如果我是,请纠正我,但这就是我认为我知道的。
- 一个请求通常会在 120 秒后超时(这是可配置的),但最终 ASP.NET 运行时会终止一个需要很长时间才能完成的请求。
- 托管环境(通常是 IIS)采用进程回收,并且可以随时决定回收您的应用程序。当这种情况发生时,所有线程都被中止并且应用程序重新启动。但是我不确定它有多激进,假设它会中止正常的正在进行的 HTTP 请求会有点愚蠢,但我希望它会中止一个线程,因为它对工作单元一无所知一个线程。
如果您必须创建一个编程模型,该模型可以轻松可靠地在理论上放置一个长时间运行的任务,并且必须运行数天,您将如何在 ASP.NET 应用程序中完成此任务?
以下是我对这个问题的看法:
我一直在考虑在 win32 服务中托管 WCF 服务。并通过 WCF 与服务对话。然而,这不是很实用,因为我选择这样做的唯一原因是从几个不同的 Web 应用程序发送任务(工作单元)。然后,我最终会要求服务提供状态更新并采取相应的行动。我对此最大的担忧是,如果我必须将每个任务部署到服务以使其能够执行一些指令,那将不是一个特别好的体验。还有这个输入问题,如果我有一个大数据集并且需要仔细研究它,我将如何为这个服务提供数据?
我现在通常做的是这个
SELECT TOP 10 *
FROM WorkItem WITH (ROWLOCK, UPDLOCK, READPAST)
WHERE WorkCompleted IS NULL
它允许我将 SQL Server 数据库用作工作队列,并使用此查询定期轮询数据库以进行工作。如果工作项目成功完成,我将其标记为已完成并继续进行,直到没有更多事情可做。我不喜欢的是理论上我可以在任何时候被打断,如果我介于成功和将其标记为完成之间,我最终可能会处理两次相同的工作项目。我可能有点偏执,这可能一切都很好,但据我所知,不能保证不会发生这种情况......
我知道以前在 SO 上也有过类似的问题,但没有给出明确的答案。这是很常见的事情,但 ASP.NET 托管环境无法处理长时间运行的工作。
请分享你的想法。