9

ここで説明されているように、MVC4の新しい非同期コントローラー機能を使用していますhttp://www.asp.net/mvc/tutorials/mvc-4/using-asynchronous-methods-in-aspnet-mvc-4

実行に10〜20秒かかる可能性のあるアクションがある場合は、進行状況をユーザーに通知するための何らかのステータスバーを提供したいと思います。非同期機能にはこれを助ける何かがありますか?

編集:私はそれをどのように試みて実行するかを突き刺し、より良い方法があるかどうかを確認します

public async Task<ActionResult> GizmosAsync()
{
    return View("Gizmos", await GetGizmosAsync());
}

private void GetGizmosAsync()
{
    for(int i=0; i<10; i++) 
    {
        lock(_locker) 
        {
           _statusMessage = String.Format("{0} of 10", i);
        }  
        DoSomethingLongRunning();
    }
}

public ActionResult Status()
{
   return Json(new { Status = _statusMessage });
}

static readonly object _locker = new object();

static string _statusMessage = "";

....

<script>

setTimeout(displayStatus, 1000);

function displayStatus() {
  $.post("/controller/status", function(data) {
    alert(data.Status);
  });
}

</script>
4

2 に答える 2

17

非同期コントローラーは、IIS の ThreadPool からスレッドを解放して、高負荷時に着信要求を処理できるようにするメカニズムにすぎませんが、クライアントとの通信は通常の要求応答のままです。

ステータスバーと並べ替えは通常、ajax リクエストが完了するまで画面に何かを表示する単なる JavaScript です。MVC4がその部分で役立つとは思いません。

次のようなことができます: https://stackoverflow.com/a/68503/1373170<div> ajax呼び出し中に「処理中...」を表示します。

編集: 実際のクライアントの進行状況と対話 (実際の進行状況など) が必要な場合は、SignalR http://www.hanselman.com/blog/AsynchronousScalableWebApplicationsWithRealtimePersistentLongrunningConnectionsWithSignalR.aspxを確認する必要があります。 また、この関連記事: Async Controllers (MVC), long running 「停止」のあるプロセス

于 2012-07-05T04:26:58.667 に答える
6

この記事は、あなたが望むものを説明しているようです:

ASP.NET MVC 3: 長時間実行されるタスクの非同期 jQuery 進行状況インジケーター

コントローラ:

public class HomeController : Controller
{
    private static IDictionary<Guid, int> tasks = new Dictionary<Guid, int>();

    public ActionResult Index()
    {
        return View();
    }

    public ActionResult Start()
    {
        var taskId = Guid.NewGuid();
        tasks.Add(taskId, 0);

        Task.Factory.StartNew(() =>
        {
            for (var i = 0; i <= 100; i++)
            {
                tasks[taskId] = i; // update task progress
                Thread.Sleep(50); // simulate long running operation
            }
            tasks.Remove(taskId);
        });

        return Json(taskId);
    }

    public ActionResult Progress(Guid id)
    {
        return Json(tasks.Keys.Contains(id) ? tasks[id] : 100);
    }
}

意見:

<script type="text/javascript">

function updateMonitor(taskId, status) {
  $("#" + taskId).html("Task [" + taskId + "]: " + status);
}

$(function () {
  $("#start").click(function (e) {
   e.preventDefault();
   $.post("Home/Start", {}, function (taskId) {

     // Init monitors
     $("#monitors").append($("<p id='" + taskId + "'/>"));
     updateMonitor(taskId, "Started");

     // Periodically update monitors
     var intervalId = setInterval(function () {
       $.post("Home/Progress", { id: taskId }, function (progress) {
         if (progress >= 100) {
           updateMonitor(taskId, "Completed");
         clearInterval(intervalId);
         } else {
           updateMonitor(taskId, progress + "%");
         }
        });
      }, 100);
    });
  });
});
</script> 
<div id="monitors"></div>
于 2014-06-05T11:31:20.130 に答える