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 ответ

Решение

Есть действительно два варианта использования

  1. Вы заботитесь о результате длительной операции
  2. Вы не заботитесь о результате (уволить и забыть)

Начнем с первого случая:

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 об асинхронных контроллерах.

Другие вопросы по тегам