ajax异步请求302分析

更新日期: 2019-03-31阅读: 6.5k标签: 请求

1.前言

遇到这样一种情况,打开网页两个窗口a,b(都是已经登录授权的),在a页面中退出登录,然后在b页面执行增删改查,这个时候因为授权原因,b页面后端的请求肯定出现异常(对这个异常的处理,进行内部跳转处理),b页面中的ajax请求的回调中就会出现问题,今天遇到了,有种恍然大悟的感觉,打开以前公司的网站发现全都没做任何处理......
没遇到过错误,永远不知道错误会什么时候出现。


2.问题:在ajax异步请求中,如果服务端出现内部跳转,如何在回调中处理

我们先来看这样一个简单的ajax异步请求

$.ajax({
    url: ‘/User/Add‘,
    type: ‘post‘,
    data: data,
    success:function(data,status)
    {
        console.log(‘data:‘+data);
        console.log(‘status:‘+status);
    },
    complete: function(xhr){
        console.log(xhr.status);
    },
    error: function (xhr) {
        debugger;
        console.log(xhr.status);
    }
});    

后端处理时,抛出异常,出现了内部重定向
稍微简单解释一下,用的mvc,加了一个全局的异常,在Add方法中抛出异常,在OnException会进行捕获到这个异常,代码很简单也比较典型。

public class GlobalHandlerErrorAttribute : HandleErrorAttribute{
     public override void OnException(ExceptionContext actionExecutedContext){
          actionExecutedContext.HttpContext.Response.Redirect("/User/Login");
    }
}
public ActionResult Add(UserModel model){
      throw new HttpException("un authorize");
}

结果是什么?
回调中会进入到success方法,输入data:是Login页面的整个内容,status:success ,完成回调complete中输出的是200,并不会和我们预期进入到error方法中来。


3.原因

用一句话来形容上面的问题就是:ajax异步请求A,A内部重定向到B。
得到的答案是:
1.ajax回调方法success中得到是B页面内容
2.ajax是用于处理数据的,并不会按照服务器的内部跳转而进行跳转,如果要跳转可以根据回调中信息进行跳转。
3.ajax请求,后端出现跳转,不会进入到error回调中的,当然回调中跳转到一个不存在的页面,那是会进入error方法中来的。
当服务器将302响应发给浏览器时,浏览器并不是直接进行ajax回调处理,而是先执行302重定向,回调success中获取的内容是重定向后的页面。
jquery源码中是这样写的,http状态码200-300或者304才会进入success回调的
isSuccess = status >= 200 && status < 300 || status === 304;//确定请求是否成功

// Cache response headers
responseHeadersString = headers || "";

// Set readyState
jqXHR.readyState = status > 0 ? 4 : 0;

// Determine if successful
isSuccess = status >= 200 && status < 300 || status === 304;

// Get response data
if ( responses ) {
    response = ajaxHandleResponses( s, jqXHR, responses );
}

// Convert no matter what (that way responseXXX fields are always set)
response = ajaxConvert( s, response, jqXHR, isSuccess );

// If successful, handle type chaining
if ( isSuccess ) {

    // Set the If-Modified-Since and/or If-None-Match header, if in ifModified mode.
    if ( s.ifModified ) {
        modified = jqXHR.getResponseHeader("Last-Modified");
        if ( modified ) {
            jQuery.lastModified[ cacheURL ] = modified;
        }
        modified = jqXHR.getResponseHeader("etag");
        if ( modified ) {
            jQuery.etag[ cacheURL ] = modified;
        }
    }

    // if no content
    if ( status === 204 || s.type === "HEAD" ) {
        statusText = "nocontent";

    // if not modified
    } else if ( status === 304 ) {
        statusText = "notmodified";

    // If we have data, let‘s convert it
    } else {
        statusText = response.state;
        success = response.data;
        error = response.error;
        isSuccess = !error;
    }
} else {
    // We extract error from statusText
    // then normalize statusText and status for non-aborts
    error = statusText;
    if ( status || !statusText ) {
        statusText = "error";
        if ( status < 0 ) {
            status = 0;
        }
    }
}


4.如何处理ajax异步请求后端跳转

ajax异步请求数据,后端出现跳转,这个时候说明程序出现了异常,可以捕获到这个异常,返回给前端,前端ajax根据返回信息,进行判断,给出正确的提示。
在全局异常中可以这样返回给前端

actionExecutedContext.Result = new JsonResult(){
    Data = new ViewModels.Web.Response.JsonResultModel(){
        code = 302,
        message = "具体的错误信息",
        success = false
    }
};

前端可以进行判断,给出相应的提示信息或者直接跳转

$.ajax({
    url: ‘/User/Add‘,
    type: ‘post‘,
    data: data,
    success:function(data,status)
    {
      if(code==302)
      {
        alert(data.message);
        //location.href=‘/User/Login‘ 也可以在返回类型加一个字段location,那就直接location.href=‘/User/Login/‘
      }
        console.log(‘data:‘+data);
        console.log(‘status:‘+status);
    },
    complete: function(xhr){
        console.log(xhr.status);
    },
    error: function (xhr) {
        debugger;
        console.log(xhr.status);
    }
});    


5.总结

1.ajax success回调处理,具体状态码为 status >= 200 && status < 300 || status === 304
2.ajax是用于异步获取数据的,并不是用于页面跳转
3.mvc中,如果给某些方法设置权限,就会导致权限验证,从而产生跳转登陆界,应该加上全局的异常捕获,既然是ajax请求,说明请求数据出现异常,应该给出相应的错误信息提示,这个提示也应该在服务器进行确定,前端负责展示。
4.ajax异步请求中,服务器不应该出现直接跳转。


链接: https://www.fly63.com/article/detial/2649

nginx 301跳转https后post请求失效问题解决

强制把http请求跳转到https,结果发现App有部分的功能不能使用,因为App一共设置了4种请求方式,分别是GET,POST,DELETE和OPTIONS方式,设置301跳转后所有的请求方法都变成了GET方式,导致一些功能无法正常使用.

HTTP请求的11个处理阶段

几乎所以有关Nginx书只要是讲深入点的就会讲到Nginx请求的11个处理阶段,要记住这些真是不易,人脑特别不擅长记住各种东西,只能做些索引罢了,能做到知道这个知识点在哪儿能找到不就行了,可是你去面试还是问这些理论,所以这里汇总下记录如下

http请求过程的7个步骤

HTTP通信机制是在一次完整的HTTP通信过程中,Web浏览器与Web服务器之间将完成下列7个步骤:建立TCP连接、Web浏览器向Web服务器发送请求命令、Web浏览器发送请求头信息、 Web服务器应答

http请求的几种类型

http请求中的8种请求方法:opions 返回服务器针对特定资源所支持的HTML请求方法 ,Get 向特定资源发出请求,Post 向指定资源提交数据进行处理请求

nodejs http请求相关总结

通过node提供的http模块,可以通过其提供的get()和request()两个方法发起http请求,get()是对request()方法的封装,方便发起get请求,如果要实现post请求,那么需要对request()方法进行封装。

POST 请求的三种常见数据提交格式

本文所讲的 POST 请求是 HTTP/1.1 协议中规定的众多 HTTP 请求方法的其中最常用的一个。一般使用 POST 请求方法向服务器发送数据(主要是一些创建更新操作),本文讨论的是 POST 请求方法常用的四种数据提交格式。

flutter之网络请求dio封装,拦截器的封装

flutter一直很火的网络请求插件dio,直接上代码,写成一个类,可以直接使用,包含请求的封装,拦截器的封装

HTTP请求过程

当我们在web浏览器的地址栏中输入: www.baidu.com,然后回车,到底发生了什么?DNS域名解析采用的是递归查询的方式,过程是,先去找DNS缓存->缓存找不到就去找根域名服务器->根域名又会去找下一级

nginx是怎么处理http请求的?

nginx首先决定要用配置文件里的哪个server{}块来处理,假设有下面的server{}配置;nginx会根据过来的http请求头里的Host字段里的值,来判断使用哪个server{}。

Js两个异步请求 同步合并数据

业务代码经常会有 两个不一样的请求,拿到数据后合并成新数组的操作。但是在异步请求中我们不知道哪个请求的回调更快返回,从而使代码的合并时间无法确定。这就需要在两个异步请求都完成后再做数据处理。

点击更多...

内容以共享、参考、研究为目的,不存在任何商业目的。其版权属原作者所有,如有侵权或违规,请与小编联系!情况属实本人将予以删除!