异步MVC控制器操作方法执行不等待等待'

时间:2014-04-03 11:18:39

标签: c# .net asp.net-mvc asp.net-mvc-4 asynchronous

我正在使用MVC 4和.NET 4.5。我希望构建一个利用新TAP(异步等待)的异步控制器。我从Controller而不是AsyncContoller继承了这个控制器,因为我使用的是基于任务的异步性,而不是基于事件的异步性。

我有两个操作方法 - 一个用于同步执行操作,另一个用于异步执行相同的操作。我的视图中的表单中还有两个提交按钮,每个操作方法一个。

以下是两种方法的代码:

同步:

[HttpPost]
public ActionResult IndexSync(FormCollection formValues)
{
        int Min = Int32.Parse(formValues["txtMin"]);
        int Count = Int32.Parse(formValues["txtCount"]);
        string Primes;
        DateTime started = DateTime.Now;
        using (BackendServiceReference.ServiceClient service = new ServiceClient())
        {
            Primes = service.GetPrimesServiceMethod(Min, Count);
        }
        DateTime ended = DateTime.Now;
        TimeSpan serviceTime = ended - started;
        ViewBag.ServiceTime = serviceTime;
        ViewBag.Primes = Primes;

        return View("Index");
}

异步:

[HttpPost]
public async Task<ActionResult> IndexAsync(FormCollection formValues)
{
        int Min = Int32.Parse(formValues["txtMin"]);
        int Count = Int32.Parse(formValues["txtCount"]);
        string Primes;
        Task<string> PrimesTask;
        DateTime started = DateTime.Now;
        using (BackendServiceReference.ServiceClient service = new ServiceClient())
        {
            PrimesTask = service.GetPrimesServiceMethodAsync(Min, Count);
        }
        DateTime ended = DateTime.Now;
        TimeSpan serviceTime = ended - started;
        ViewBag.ServiceTime = serviceTime;
        Primes = await PrimesTask;
        ViewBag.Primes = Primes;

        return View("Index");
}

在async方法中,我希望在调用服务方法后立即执行DateTime ended = DateTime.Now,而耗时的服务方法在后台异步执行。

但是,这不会发生,并且执行等待&#39;调用服务方法时,而不是等待Primes = await PrimesTask发生的地方。

我有什么遗漏的吗?

赞赏正确的方向。

1 个答案:

答案 0 :(得分:4)

我怀疑它实际上是ServiceClient.Dispose中的阻止。

要解决此问题,请展开using块以包含您的await

using (BackendServiceReference.ServiceClient service = new ServiceClient())
{
    PrimesTask = service.GetPrimesServiceMethodAsync(Min, Count);
    DateTime ended = DateTime.Now;
    TimeSpan serviceTime = ended - started;
    ViewBag.ServiceTime = serviceTime;
    Primes = await PrimesTask;
}
ViewBag.Primes = Primes;