Вы можете использовать длинный опрос со скрытым iframe и частичной кодировкой передачи с сервера, который будет выплевывать теги <script>
по мере поступления данных.В этом теге сценария вы можете вызывать пользовательскую функцию JavaScript обратного вызова, которая позаботится о форматировании результатов.
ОБНОВЛЕНИЕ:
В соответствии с запросом в разделе комментариев, здесь приведен пример реализацииметод длинного опроса с использованием скрытого фрейма.
Предположим, у вас есть какая-то модель:
public class MyViewModel
{
public string Foo { get; set; }
}
и у вас есть служба, которая возвращает эту модель в чанках и уведомляет вызывающую сторону, что чанк доступен с помощью событий:
public class MyService
{
public void GetModels(Action<MyViewModel, object> onModelAvailable, object state, Action onComplete)
{
Task.Factory.StartNew(x =>
{
try
{
for (int i = 0; i < 10; i++)
{
onModelAvailable(new MyViewModel
{
Foo = "foo " + i
}, x);
Thread.Sleep(1000);
}
}
finally
{
onComplete();
}
}, state);
}
}
Теперь у нас может быть следующий контроллер:
public class HomeController : AsyncController
{
public ActionResult Index()
{
return View();
}
public ActionResult LongPoll()
{
var service = new MyService();
return new MyActionResult(service);
}
}
и следующий вид:
<script type="text/javascript">
// we define a callback function which will be invoked
// when a chunk is available from the server
var callback = function (model) {
// the model variable passed here will represent the chunk
$($('<div/>', {
html: model.Foo
})).appendTo('#result');
};
</script>
<iframe style="display:none;" src="@Url.Action("longpoll")"></iframe>
<div id="result"></div>
Теперь последняя часть курса - это реализация пользовательскогорезультат действия, которое выполнит фрагментарный перевод:
public class MyActionResult : ActionResult
{
private readonly MyService _service;
public MyActionResult(MyService service)
{
_service = service;
}
public override void ExecuteResult(ControllerContext context)
{
var response = context.HttpContext.Response;
response.BufferOutput = true;
response.ContentType = "text/html";
var wait = new ManualResetEvent(false);
_service.GetModels((model, state) =>
{
var httpResponse = (HttpResponseBase)state;
httpResponse.BufferOutput = true;
httpResponse.ContentType = "text/html";
var serializer = new JavaScriptSerializer();
var script = string.Format(
"<script type=\"text/javascript\">window.parent.callback({0});</script>",
serializer.Serialize(model)
);
httpResponse.Write(script);
httpResponse.Flush();
},
response,
() =>
{
wait.Set();
});
wait.WaitOne();
}
}