我想停止服务器端jQuery.ajax
方法调用的操作。
我可以在客户端使用$.ajax.abort()
方法停止Ajax请求,但不能在服务器端停止。
更新
我使用异步操作而不是同步操作,但我没有得到我想要的!如您所知,服务器无法同时处理多个请求,导致每个请求必须等到上一个请求完成,即使先前的请求被$ .Ajax.Abort()方法取消。 我知道如果我使用[SessionState(System.Web.SessionState.SessionStateBehavior.ReadOnly)]属性,它几乎是我想要的,但它不能让我满意。
最重要的是我想在用户端中止服务器端的处理方法。就是这样:))
答案 0 :(得分:4)
您可能希望使用以下类型的控制器Using an Asynchronous Controller in ASP.NET MVC
并且还看看这篇文章是否也帮助了你Cancel async web service calls,抱歉这次我无法提供任何代码示例。
我创建了一个示例作为概念证明,表明您可以取消服务器端请求。 My github async cancel example
如果您通过代码调用其他网站,则有两种选择,具体取决于您的目标框架以及您要使用的方法。我在这里提供了参考资料供您审阅:
WebRequest.BeginGetResponse用于.Net 4.0 {。3}}用于.Net 4.5,此类有一个取消所有待处理请求的方法。
希望这能为您提供足够的信息来实现目标。
答案 1 :(得分:3)
这是一个例子 后端:
[HttpGet]
public List<SomeEntity> Get(){
var gotResult = false;
var result = new List<SomeEntity>();
var tokenSource2 = new CancellationTokenSource();
CancellationToken ct = tokenSource2.Token;
Task.Factory.StartNew(() =>
{
// Do something with cancelation token to break current operation
result = SomeWhere.GetSomethingReallySlow();
gotResult = true;
}, ct);
while (!gotResult)
{
// When you call abort Response.IsClientConnected will = false
if (!Response.IsClientConnected)
{
tokenSource2.Cancel();
return result;
}
Thread.Sleep(100);
}
return result;
}
使用Javascript:
var promise = $.post("/Somewhere")
setTimeout(function(){promise.abort()}, 1000)
希望我不要迟到。
答案 2 :(得分:0)
我们已经在IE中看到了问题,其中中止的请求仍然被转发到控制器操作 - 但是,在我们的日志和用户活动条目中报告了剥离的参数,这导致了不同的错误。
我使用类似以下的过滤器解决了这个问题
using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;
using System.Web.Mvc;
using TWV.Infrastructure;
using TWV.Models;
namespace TWV.Controllers
{
public class TerminateOnAbortAttribute : FilterAttribute, IActionFilter
{
public void OnActionExecuting(ActionExecutingContext filterContext)
{
// IE does not always terminate when ajax request has been aborted - however, the input stream gets wiped
// The content length - stays the same, and thus we can determine if the request has been aborted
long contentLength = filterContext.HttpContext.Request.ContentLength;
long inputLength = filterContext.HttpContext.Request.InputStream.Length;
bool isAborted = contentLength > 0 && inputLength == 0;
if (isAborted)
{
filterContext.Result = new EmptyResult();
}
}
public void OnActionExecuted(ActionExecutedContext filterContext)
{
// Do nothing
}
}
}
答案 3 :(得分:0)
我发现这篇文章:Cancelling Long Running Queries in ASP.NET MVC and Web API 非常有帮助。我将在这里总结这篇文章。
开始前的速记:
abort
取消操作。我最终得到了这样的结果:
// Just add the CancellationToken parameter to the end of your parameter list
public async Task<ActionResult> GetQueryDataAsync(
int id,
CancellationToken cancellationToken)
{
CancellationToken dcToken = Response.ClientDisconnectedToken;
var linkTokenSrc = CancellationTokenSource
.CreateLinkedTokenSource(
cancellationToken,
dcToken
);
using (var dbContext = new Entities())
{
var data = await dbContext.ComplicatedView
/* A bunch of linq statements */
.ToListAsync(linkTokenSrc.Token)
.ConfigureAwait(true);
// Do stuff and return data unless cancelled
// ...
}
}
由于我使用的是 MVC5,我不得不使用文章中规定的解决方法,即从 Response.ClientDisconnectedToken
读取并链接令牌源。
如果您使用的是 ASP.NET Core MVC,您应该可以直接使用 CancellationToken
参数,而不必根据文章进行链接。
此后,我所要做的就是确保在离开页面时abort
在 JavaScript 中设置了 XHR。
答案 4 :(得分:-1)
一个简单的解决方案是使用类似下面的内容。我用它来取消长时间运行的任务(特别是生成数千个通知)。我也使用相同的方法来轮询进度并通过AJAX更新进度条。此外,这几乎适用于任何版本的MVC,并不依赖于.NET的新功能
public class MyController : Controller
{
private static m_CancelAction = false;
public string CancelAction()
{
m_CancelAction = true;
return "ok";
}
public string LongRunningAction()
{
while(...)
{
Dosomething (i.e. Send email, notification, write to file, etc)
if(m_CancelAction)
{
m_CancelAction = false;
break;
return "aborted";
}
}
return "ok";
}
}