• 企业400电话
  • 微网小程序
  • AI电话机器人
  • 电商代运营
  • 全 部 栏 目

    企业400电话 网络优化推广 AI电话机器人 呼叫中心 网站建设 商标✡知产 微网小程序 电商运营 彩铃•短信 增值拓展业务
    关于Ajax跨域问题及解决方案详析

    复现Ajax跨域问题

    做两个简单的小项目复现Ajax跨域问题. 后端语言使用Java

    首先是一个简单的订单系统, 通过访问/loadOrderList, 最终以json串形式返回订单集合. 该项目使用Tomcat发布在7070端口.

    @RequestMapping("/loadOrderList")
    @ResponseBody
    public ListOrder> loadOrderList(String uid){
     //模拟订单数据
     Order o1 = new Order();
     o1.setId("111");
     o1.setTotal(333.33);
     o1.setDate("2019-4-29");
    
     Order o2 = new Order();
     o2.setId("222");
     o2.setTotal(444.44);
     o2.setDate("2019-5-29");
    
     Order o3 = new Order();
     o3.setId("333");
     o3.setTotal(555.55);
     o3.setDate("2019-6-29");
    
     ListOrder> list = new ArrayList>();
     list.add(o1);
     list.add(o2);
     list.add(o3);
    
     return list;
    }

    在另一个项目中做一个向订单系统发送一个ajax请求, 获取订单集合. 该项目使用Tomcat插件发布在9090端口.

    //index.jsp
    
    %@ page contentType="text/html;charset=UTF-8" language="java" %>
    html>
    head>
     title>Title/title>
     script type="text/javascript" src="https://code.jquery.com/jquery-1.11.3.js">/script>
     script type="text/javascript">
     function sendAjax() {
      $.post("http://localhost:7070/order/loadOrderList", "uid=1234", function (data) {
      alert(data);
      });
     }
     /script>
    /head>
    body>
     a href="javascript:sendAjax()" rel="external nofollow" rel="external nofollow" >sendAjax/a>
    /body>
    /html>

    点击sendAjax超链接向订单系统发送ajax请求.

    通过开发者工具发现虽然服务器以状态码200响应回来, 但是控制台却报错了.

    这就是Ajax跨域出错的一种表现, 下面分析原因.

    Ajax跨域介绍

    Accept: */*
    Accept-Encoding: gzip, deflate, br
    Accept-Language: zh-CN,zh;q=0.9
    Connection: keep-alive
    Content-Length: 8
    Content-Type: application/x-www-form-urlencoded; charset=UTF-8
    Host: localhost:7070
    Origin: http://localhost:9090 协议+域名+端口

    Ajax跨域解决方案

    下面介绍最常用Ajax跨域解决方案.

    一. 在服务端添加响应头Access-Control-Allow-Origin

    dependency>
     groupId>com.thetransactioncompany/groupId>
     artifactId>cors-filter/artifactId>
     version>2.5/version>
     scope>runtime/scope>
    /dependency>
     filter>
     filter-name>CORS/filter-name>
     filter-class>com.thetransactioncompany.cors.CORSFilter/filter-class>
     init-param>
      param-name>cors.allowOrigin/param-name>!--这个标签是关键, *代表所有源都能访问-->
      param-value>*/param-value>
     /init-param>
     init-param>
      param-name>cors.supportedMethods/param-name>
      param-value>GET, POST, HEAD, PUT, DELETE/param-value>
     /init-param>
     init-param>
      param-name>cors.supportedHeaders/param-name>
      param-value>Accept, Origin, X-Requested-With, Content-Type, Last-Modified/param-value>
     /init-param>
     init-param>
      param-name>cors.exposedHeaders/param-name>
      param-value>Set-Cookie/param-value>
     /init-param>
     init-param>
      param-name>cors.supportsCredentials/param-name>
      param-value>true/param-value>
     /init-param>
     /filter>
    
     filter-mapping>
     filter-name>CORS/filter-name>
     url-pattern>/*/url-pattern>
     /filter-mapping>

    配置后重启订单项目, 再次发起Ajax请求可以看到成功返回数据, 响应头中包含了Access-Control-Allow-Origin, 值为发起Ajax请求的源.



     二. 使用JSONP解决

    %@ page contentType="text/html;charset=UTF-8" language="java" %>
    html>
    head>
     title>Title/title>
     script type="text/javascript" src="https://code.jquery.com/jquery-1.11.3.js">/script>
     script>
      function doCallBack(data){
       var str = JSON.stringify(data);
       alert(str);
      }
     /script>
    /head>
    body>
     script src="http://localhost:7070/order/loadOrderList3?uid=111callBack=doCallBack">/script>
    /body>
    /html>
    @RequestMapping("/loadOrderList3")
    @ResponseBody
    public String loadOrderList3(String uid, String callBack){
     //模拟订单数据
     Order o1 = new Order();
     o1.setId("111");
     o1.setTotal(333.33);
     o1.setDate("2019-4-29");
    
     Order o2 = new Order();
     o2.setId("222");
     o2.setTotal(444.44);
     o2.setDate("2019-5-29");
    
     Order o3 = new Order();
     o3.setId("333");
     o3.setTotal(555.55);
     o3.setDate("2019-6-29");
    
     ListOrder> list = new ArrayList>();
     list.add(o1);
     list.add(o2);
     list.add(o3);
    
     //拼接js代码
     String result = callBack + "(" + JSON.toJSONString(list) + ")";
     return result;
    }

    这个想法是不是很妙? 明白这个原理之后, 我们可以使用jQuery方便进行JSONP操作, 在上面的代码中我们人为指定了一个名为doCallBack的函数, 而jQuery会随机用时间戳生成一个函数名, 原理和上面是一样的.

    所以完成一开时点击超链接发送Ajax请求只需要如下几步.

    %@ page contentType="text/html;charset=UTF-8" language="java" %>
    html>
    head>
     title>Title/title>
     script type="text/javascript" src="https://code.jquery.com/jquery-1.11.3.js">/script>
     script>
      function sendAjax(){
       $.getJSON("http://localhost:7070/order/loadOrderList3?callBack=?","uid=111",
       function (data) {
        var str = JSON.stringify(data);
        alert(str);
       });
      }
     /script>
    /head>
    body>
     a href="javascript:sendAjax()">sendAjax/a>
    /body>
    /html>

    小结

    上面两种解决办法在思路上有着本质的不同. 方案一抓住CORS跨域访问问题的本质, 在后端加上响应头解决跨域问题. 方案二JSONP利用的是script>标签能够跨域获取js代码的特性, 绕过跨域问题.

    总结

    以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对脚本之家的支持。

    您可能感兴趣的文章:
    • 详解ajax跨域问题解决方案
    • Ajax跨域问题及解决方案(jsonp,cors)
    • 解决前端跨域问题方案汇总
    • ajax请求前端跨域问题原因及解决方案
    上一篇:Ajax登陆使用Spring Security缓存跳转到登陆前的链接
    下一篇:Ajax请求跨域问题解决方案分析
  • 相关文章
  • 

    © 2016-2020 巨人网络通讯 版权所有

    《增值电信业务经营许可证》 苏ICP备15040257号-8

    关于Ajax跨域问题及解决方案详析 关于,Ajax,跨域,问题,及,