转载自  搞定所有的跨域请求问题

网上各种跨域教程,各种实践,各种问答,除了简单的 jsonp 以外,很多说 CORS 的都是行不通的,老是缺那么一两个关键的配置。本文只想解决问题,所有的代码经过亲自实践。

本文解决跨域中的 get、post、data、cookie 等这些问题。

本文只会说 get 请求和 post 请求,读者请把 post 请求理解成除 get 请求外的所有其他请求方式。

JSONP

jsonp 的原理很简单,利用了【前端请求静态资源的时候不存在跨域问题】这个思路。

但是 只支持 get,只支持 get,只支持 get

注意一点,既然这个方法叫 jsonp,后端数据一定要使用 json 数据,不能随便的搞个字符串什么的,不然你会觉得结果莫名其妙的。

前端 jQuery 写法

$.ajax({type: "get",url: baseUrl + "/jsonp/get",dataType: "jsonp",success: function(response) {$("#response").val(JSON.stringify(response));}
});

dataType: "jsonp"。除了这个,其他配置和普通的请求是一样的。

后端 SpringMVC 配置

如果你也使用 SpringMVC,那么配置一个 jsonp 的 Advice 就可以了,这样我们写的每一个 Controller 方法就完全不需要考虑客户端到底是不是 jsonp 请求了,Spring 会自动做相应的处理。

@ControllerAdvice
public class JsonpAdvice extends AbstractJsonpResponseBodyAdvice {public JsonpAdvice(){// 这样如果请求中带 callback 参数,Spring 就知道这个是 jsonp 的请求了super("callback");}
}

以上写法要求 SpringMVC 版本不低于 3.2,低于 3.2 的我只能说,你们该升级了。

后端非 SpringMVC 配置

以前刚工作的时候,Struts2 还红遍天,几年的光景,SpringMVC 就基本统治下来了国内市场。

偷懒一下,这里贴个伪代码吧,在我们的方法返回前端之前调一下 wrap 方法:

public Object wrap(HttpServletRequest request){String callback = request.getParameter("callback");if(StringUtils.isBlank(callback)){return result;} else {return callback+"("+JSON.toJSONString(result)+")";}
}

CORS

Cross-Origin Resource Sharing

毕竟 jsonp 只支持 get 请求,肯定不能满足我们的所有的请求需要,所以才需要搬出 CORS。

国内的 web 开发者还是比较苦逼的,用户死不升级浏览器,老板还死要开发者做兼容。

CORS 支持以下浏览器,目前来看,浏览器的问题已经越来越不重要了,连淘宝都不支持 IE7 了~

  • Chrome 3+
  • Firefox 3.5+
  • Opera 12+
  • Safari 4+
  • Internet Explorer 8+

前端 jQuery 写法

直接看代码吧:

$.ajax({type: "POST",url: baseUrl + "/jsonp/post",dataType: 'json',crossDomain: true,xhrFields: {withCredentials: true},data: {name: "name_from_frontend"},success: function (response) {console.log(response)// 返回的 json 数据$("#response").val(JSON.stringify(response));}
});

dataType: "json",这里是 json,不是 jsonp,不是 jsonp,不是 jsonp。

crossDomain: true,这里代表使用跨域请求

xhrFields: {withCredentials: true},这样配置就可以把 cookie 带过去了,不然我们连 session 都没法维护,很多人都栽在这里。当然,如果你没有这个需求,也就不需要配置这个了。

后端 SpringMVC 配置

对于大部分的 web 项目,一般都会有 mvc 相关的配置类,此类继承自 WebMvcConfigurerAdapter。如果你也使用 SpringMVC 4.2 以上的版本的话,直接像下面这样添加这个方法就可以了:

@Configuration
public class WebConfig extends WebMvcConfigurerAdapter {@Overridepublic void addCorsMappings(CorsRegistry registry) {registry.addMapping("/**/*").allowedOrigins("*");}
}

如果很不幸你的项目中 SpringMVC 版本低于 4.2,那么需要「曲线救国」一下:

public class CrossDomainFilter extends OncePerRequestFilter {@Overrideprotected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain filterChain) throws ServletException, IOException {response.addHeader("Access-Control-Allow-Origin", "*");// 如果提示 * 不行,请往下看response.addHeader("Access-Control-Allow-Credentials", "true");response.addHeader("Access-Control-Allow-Methods", "GET, POST, PUT, DELETE");response.addHeader("Access-Control-Allow-Headers", "Content-Type");filterChain.doFilter(request, response);}
}

在 web.xml 中配置下 filter:

<filter><filter-name>CrossDomainFilter</filter-name><filter-class>com.javadoop.filters.CrossDomainFilter</filter-class>
</filter>
<filter-mapping><filter-name>CrossDomainFilter</filter-name><url-pattern>/*</url-pattern>
</filter-mapping>

有很多项目用 shiro 的,也可以通过配置 shiro 过滤器的方式,这里就不介绍了。

注意了,我说的是很笼统的配置,对于大部分项目是可以这么笼统地配置的。文中类似 “*” 这种配置读者应该都能知道怎么配。

如果读者发现浏览器提示不能用 ‘*’ 符号,那读者可以在上面的 filter 中根据 request 对象拿到请求头中的 referer(request.getHeader("referer")),然后动态地设置 "Access-Control-Allow-Origin":

String referer = request.getHeader("referer");
if (StringUtils.isNotBlank(referer)) {URL url = new URL(referer);String origin = url.getProtocol() + "://" + url.getHost();response.addHeader("Access-Control-Allow-Origin", origin);
} else {response.addHeader("Access-Control-Allow-Origin", "*");
}

2018-04-28:今天终于知道为什么有时候会提示我们 * 不支持了,原来是只要前端写了 withCredentials: true 那么浏览器就会提示这个,一种办法就是这里说的使用动态构造 origin 的方式,另一种办法就是跨域不传 cookie,让前端把 cookie 要传的信息(如 sessionId/accessKey) 放到 header 中或者直接写在 request 的参数里。

前端非 jQuery 写法

jQuery 一招鲜吃遍天的日子是彻底不在了,这里就说说如果不使用 jQuery 的话,怎么解决 post 跨域的问题。大部分的 js 库都会提供相应的方案的,大家直接找相应的文档看看就知道怎么用了。

来一段原生 js 介绍下:

function createCORSRequest(method, url) {var xhr = new XMLHttpRequest();if ("withCredentials" in xhr) {// 如果有 withCredentials 这个属性,那么可以肯定是 XMLHTTPRequest2 对象。看第三个参数xhr.open(method, url, true);} else if (typeof XDomainRequest != "undefined") {// 此对象是 IE 用来跨域请求的xhr = new XDomainRequest();xhr.open(method, url);} else {// 如果是这样,很不幸,浏览器不支持 CORSxhr = null;}return xhr;
}var xhr = createCORSRequest('GET', url);
if (!xhr) {throw new Error('CORS not supported');
}

其中,Chrome,Firefox,Opera,Safari 这些「程序员友好」的浏览器使用的是 XMLHTTPRequest2 对象。IE 使用的是 XDomainRequest。

搞定所有的跨域请求问题相关推荐

  1. 轻松搞定JSONP跨域请求

    轻松搞定JSONP跨域请求 一.同源策略 要理解跨域,先要了解一下"同源策略".所谓同源是指,域名,协议,端口相同.所谓"同源策略",简单的说就是基于安全考虑, ...

  2. 利用Access-Control-Allow-Origin响应头解决跨域请求

    标签:cors 跨域请求 ACAO 2015-11-28 18:53 12103人阅读 评论(0) 收藏 举报  分类: web前端(5)  J2EE服务器(2)  传统的跨域请求没有好的解决方案,无 ...

  3. Jquery跨域请求php数据(jsonp)

    Jquery跨域请求php数据 我们一般用到ajax的时候是在同服务器下,一般情况下不会跨域,但有时候需要调用其他域名或ip下的数据的时候,遇到跨域请求数据的时候. 今天在工作中碰到javascrip ...

  4. Cors跨域(一):深入理解跨域请求概念及其根因

    Talk is cheap. Show me the money. 前言 你好,我是YourBatman. 做Web开发的小伙伴对"跨域"定并不陌生,像狗皮膏药一样粘着几乎每位同学 ...

  5. vue-resource ajax跨域,ajax 跨域请求 vue-resource jsonp跨域

    aj体朋几一级发等点确层数框的很屏果行4带域ax 跨域请求 vue-resource j直分调浏器代,刚求的一学础过功互有解小久宗点差维含数如数围请sonp跨域 最近在学习vue.js 碰到个ajax ...

  6. springboot实现ajax跨域请求

    有段时间没写文章了.看到有人提问ajax跨域请求的问题. 博主要再次强调,跨域,就是从不同的的IP端口获取数据,比如说,从www.baidu.com获取数据,就叫跨域!那么localhost:8080 ...

  7. 【漏洞利用】JSONP跨域请求漏洞 挖掘、利用详解

    参考文章 轻松搞定JSONP跨域请求 jsonp跨域原理,使用以及同源策略 跨域漏洞丨JSONP和CORS跨域资源共享 Tag: Ref: [[019.同源策略]] 本片文章仅供学习使用,切勿触犯法律 ...

  8. 服务器为C#开发,aspx设置跨域请求

    aspx设置跨域请求 最近想用hbuilder把项目转成app,要请求之前的服务器,所以遇到个跨域问题,查了很多资料终于搞定了... 在服务器aspx页面头文件里加: <meta http-eq ...

  9. ajax 跨域请求,每次会发送两个请求?

    2019独角兽企业重金招聘Python工程师标准>>> 跨域已经是个老话题了,但是最近搞百度的语音接口的时候,在服务端配置了 CORS ,跨域倒是没问题,但是每次都会发送两个请求: ...

最新文章

  1. AXI SG DMA 简易驱动 版本构思 (一)
  2. Xcode10 闪退问题
  3. mysql 统计本月的_mysql 查询当天、本周,本月,上一个月的数据
  4. 硬盘安装Linux救援系统,硬盘安装linux系统
  5. [转载]C/C++框架和库
  6. arm linux 中断 分析,armlinux中断异常的处理分析.pdf
  7. c++中unordered_map的坑
  8. 反函数连续性定理 反三角_高中数学:三角函数诱导公式及诱导公式口诀
  9. 【opencv学习】Blob检测斑点
  10. 可执行镜像——开发环境的Docker化之路
  11. GNU make manual 翻译(三十八)
  12. java 显示锁_Java 实现一个自己的显式锁Lock(有超时功能)
  13. Uva 1471 Defense Lines(LIS变形)
  14. mongo性能测试demo 代码正确运行
  15. matlab读取sgy格式文件的m文件,matlab读取segy格式的文件
  16. 免费后台管理UI界面、html源码推荐
  17. 使用prometheus(普罗米修斯)监控redis容器详解
  18. STM8L051的硬件I2C调试
  19. 往事不回头,今后不将就~~Fighting
  20. OHIF记录(二)——Viewers和React-vtk工具包互联

热门文章

  1. 计算机科普小知识——Win7系统32位与64位的区别,该如何选择?
  2. 软件构造学习笔记-实验4
  3. 栈的基础概念与经典题目(Leetcode题解-Python语言)
  4. [MyBatisPlus]代码生成器
  5. [Java基础]反射获取成员变量并使用练习
  6. [C++11]final关键字的使用
  7. 利用链表实现可合并堆(算法导论第三版思考题10-2)
  8. 给定一个n节点的二叉树,写出一个O(n)时间递归过程,将该树每个节点关键字输出(算法导论第十章10.4-2)
  9. 番茄时间有感之关于在疫情期间我与ACM不得不说的故事
  10. Collecting Bugs POJ - 2096(基础概率dp+期望模板)