我有一个动作,发送一个简单的电子邮件:
[HttpPost,ActionName("Index")] public ActionResult IndexPost(ContactForm contactForm) { if (ModelState.IsValid) { new EmailService().SendAsync(contactForm.Email,contactForm.Name,contactForm.Subject,contactForm.Body,true); return RedirectToAction(MVC.Contact.Success()); } return View(contactForm); }
和电子邮件服务:
public void SendAsync(string fromEmail,string fromName,string subject,string body,bool isBodyHtml) { MailMessage mailMessage.... .... SmtpClient client = new SmtpClient(settingRepository.SmtpAddress,settingRepository.SmtpPort); client.EnableSsl = settingRepository.SmtpSsl; client.Credentials = new NetworkCredential(settingRepository.SmtpUserName,settingRepository.SmtpPassword); client.SendCompleted += client_SendCompleted; client.SendAsync(mailMessage,Tuple.Create(client,mailMessage)); } private void client_SendCompleted(object sender,System.ComponentModel.AsyncCompletedEventArgs e) { Tuple<SmtpClient,MailMessage> data = (Tuple<SmtpClient,MailMessage>)e.UserState; data.Item1.Dispose(); data.Item2.Dispose(); if (e.Error != null) { } }
当我发送电子邮件时,我使用Async方法,然后我的方法SendAsync立即返回,然后调用RedirectToAction。但是ASP.NET不会发送响应(在这种情况下是重定向),直到client_SendCompleted完成。
以下是我想要了解的内容:
当在Visual Studio调试器中观察执行时,SendAsync会立即返回(并且RedirectToAction被调用),但在发送电子邮件之前,浏览器中没有任何反应?
如果我在client_SendCompleted里面放了一个断点,客户端会继续加载….直到我在调试器上点击F5。
解决方法
这是设计。如果异步工作以调用基础SynchronizationContext的方式被启动,ASP.NET将自动等待任何未完成的异步工作完成请求之前完成。这是为了确保如果您的异步操作尝试与HttpContext,HttpResponse等进行交互,它将仍然在周围。
如果你想做真火忘了,你需要在ThreadPool.QueueUserWorkItem中包装你的调用。这将强制它运行在一个新的线程池线程,而不需要通过SynchronizationContext,所以请求将愉快地返回。
请注意,如果由于任何原因,当您的发送仍在进行中时,应用程式域名将会下降(例如,如果您更改了web.config文件,将新文件放入bin,应用程式池回收等),您的异步发送将突然中断。如果你关心这一点,看看Phil Haacks WebBackgrounder for ASP.NET,它让你排队和运行后台工作(如发送电子邮件),以确保在应用程序域关闭的情况下优雅地完成。