ASP.NET MVC2 Вызовите метод AsyncController из jQuery? - PullRequest
2 голосов
/ 20 февраля 2011

Я пытаюсь научиться использовать 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)
    {            
    }
}

, а затем вызываю метод Export из jQuery:

function Export() {
    $.post("Background/Export", { id: $("#Id").val() }, function (data) {
    // nothing to do yet
    });
}

НО в результате возникает ошибка 404 (не найдено)не найден).Если я пытаюсь вызвать Background / Hello или Background / ExportAysnc, они найдены.

Что я делаю не так?

1 Ответ

8 голосов
/ 20 февраля 2011

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

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

...