ASP.NET MVC2 Вызовите метод AsyncController из jQuery?
Я пытаюсь научиться использовать AsyncController в MVC2, но там очень мало документации / руководств. Я собираюсь взять один обычный метод контроллера, который имеет очень медленный экспорт в стороннюю службу, и преобразовать его в асинхронный метод.
Оригинальный метод контроллера:
public JsonResult SaveSalesInvoice(SalesInvoice invoice)
{
SaveInvoiceToDatabase(invoice); // this is very quick
ExportTo3rdParty(invoice); // this is very slow and should be async
}
Поэтому я создал новый контроллер, который наследуется от AsyncController:
public class BackgroundController : AsyncController
{
public void ExportAysnc(int id)
{
SalesInvoice invoice = _salesService.GetById(id);
ExportTo3rdParty(invoice);
}
public void ExportCompleted(int id)
{
// I dont care about the return value right now,
// because the ExportTo3rdParty() method
// logs the result to a table
}
public void Hello(int id)
{
}
}
А затем вызовите метод экспорта из jQuery:
function Export() {
$.post("Background/Export", { id: $("#Id").val() }, function (data) {
// nothing to do yet
});
}
НО результатом является ошибка 404 not found (фон / экспорт не найден). Если я пытаюсь вызвать Background / Hello или Background/ExportAysnc, они найдены.
Что я делаю неправильно?
1 ответ
Есть действительно два варианта использования
- Вы заботитесь о результате длительной операции
- Вы не заботитесь о результате (уволить и забыть)
Начнем с первого случая:
public class BackgroundController : AsyncController
{
public void ExportAysnc(int id)
{
AsyncManager.OutstandingOperations.Increment();
Task.Factory.StartNew(() => DoLengthyOperation(id));
// Remark: if you don't use .NET 4.0 and the TPL
// you could manually start a new thread to do the job
}
public ActionResult ExportCompleted(SomeResult result)
{
return Json(result, JsonRequestBehavior.AllowGet);
}
private void DoLengthyOperation(int id)
{
// TODO: Make sure you handle exceptions here
// and ensure that you always call the AsyncManager.OutstandingOperations.Decrement()
// method at the end
SalesInvoice invoice = _salesService.GetById(id);
AsyncManager.Parameters["result"] = ExportTo3rdParty(invoice);
AsyncManager.OutstandingOperations.Decrement();
}
}
Теперь вы можете вызвать это так:
$.getJSON(
'<%= Url.Action("Export", "Background") %>',
{ id: $("#Id").val() },
function (data) {
// do something with the results
}
);
Теперь, когда вы упомянули вызов веб-службы, это означает, что при создании клиентского прокси веб-службы у вас была возможность испускать асинхронные методы (XXXCompleted и XXXAsync):
public class BackgroundController : AsyncController
{
public void ExportAysnc(int id)
{
AsyncManager.OutstandingOperations.Increment();
// that's the web service client proxy that should
// contain the async versions of the methods
var someService = new SomeService();
someService.ExportTo3rdPartyCompleted += (sender, e) =>
{
// TODO: Make sure you handle exceptions here
// and ensure that you always call the AsyncManager.OutstandingOperations.Decrement()
// method at the end
AsyncManager.Parameters["result"] = e.Value;
AsyncManager.OutstandingOperations.Decrement();
};
var invoice = _salesService.GetById(id);
someService.ExportTo3rdPartyAsync(invoice);
}
public ActionResult ExportCompleted(SomeResult result)
{
return Json(result, JsonRequestBehavior.AllowGet);
}
}
Это наилучшее возможное использование асинхронного контроллера, поскольку он опирается на порты завершения ввода / вывода и не монополизирует какие-либо потоки на сервере во время выполнения длительной операции.
Второй случай проще (на самом деле не нужен асинхронный контроллер):
public class BackgroundController : Controller
{
public ActionResult Export(int id)
{
// Fire and forget some lengthy operation
Task.Factory.StartNew(() => DoLengthyOperation(id));
// return immediately
return Json(new { success = true }, JsonRequestBehavior.AllowGet);
}
}
Вот хорошая статья на MSDN об асинхронных контроллерах.