SpringBoot解決跨域的超實用方案分享
一、驗證跨域
模擬前端地址:http://localhost:8080/index.html
后端接口地址:http://localhost:8081/auth/test/cors
思路:啟動兩個springboot項目:8080端口和8081端口,在8080端口的靜態(tài)資源目錄index.html中,對8081端口的接口發(fā)起請求,因為端口不同,所以屬于跨域,從而就可以驗證不同請求(
POST或OPTIONS等方法)是否報CORS error !
1. 添加index.html
在 tg-book-web的resources下,新建static\index.html
邏輯:使用ajax發(fā)送請求,調(diào)用8081的/auth/test/cors接口,如果執(zhí)行成功,將結(jié)果輸出到id="result"的div下,如下代碼是post請求,后面還會改成options請求測試.
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
<script type="text/javascript" src="https://code.jquery.com/jquery-3.1.1.min.js"></script>
</head>
<body>
<div id="result">
</div>
</body>
<script type="text/javascript">
$.ajax({
url: 'http://localhost:8081/auth/test/cors',
method: 'post',
success: function (res) {
console.log(res)
$('#result').append('<p>' + res['message'] + '</p>')
}
})
</script>
</html>對index.html不進行接口身份認(rèn)證
因為我們增加了身份認(rèn)證AuthInterceptor,所以需要在AuthInterceptor的preHandle過濾掉index.html,如下:
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
System.out.println("AuthInterceptor.preHandle:" + request.getServletPath());
if (request.getServletPath().startsWith("/index.html")) {
return true;
}
...其它代碼
}2. 增加/auth/test/cors接口
AuthController下增加接口
@PostMapping("/test/cors")
public TgResult<String> testCors() {
return TgResult.ok();
}同樣,將/auth/test/cors也不進行接口身份認(rèn)證.
排除身份認(rèn)證是為了更好的測試跨域,所以同樣寫在排除"/auth/login"的位置.
registry.addInterceptor(authInterceptor)
.addPathPatterns("/**")
// 排除的請求路徑
.excludePathPatterns("/auth/login", "/auth/test/cors");3. IDEA啟動多個springboot項目
關(guān)于如何使用IDEA啟動多個springboot項目,我就不演示了,我啟動了8080和8081兩個端口:

4. 驗證POST方法
在瀏覽器打開 http://localhost:8080/index.html,如下圖:

可以看到,8080/index.html通過ajax請求了8081的接口,返回200,說明支持POST請求跨域!
5. 驗證OPTIONS方法
OK,正如我開頭所說,@CrossOrigin注解對options方法并沒有支持跨域,我們來試一試!
只需將index.html中的 【 method: ‘post’】改為【 method: ‘options’】,重啟8080服務(wù),然后用同樣的方法測試,如下,報CORS error了 ??

詳細(xì)報錯:

二、攔截器方案
SpringBoot中使用攔截器需要2步:
- 定義攔截器
- 注冊攔截器,并指定攔截規(guī)則
定義攔截器
新增一個攔截器類 CorsInterceptor 實現(xiàn) HandlerInterceptor 接口
package org.tg.book.web.interceptor;
@Component
public class CorsInterceptor implements HandlerInterceptor {
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) {
// 允許跨域源,全部允許為*,否則寫允許請求的源ip
if (StringUtils.isEmpty(request.getHeader("Origin"))) {
response.setHeader("Access-Control-Allow-Origin", "*");
} else {
response.setHeader("Access-Control-Allow-Origin", request.getHeader("Origin"));
}
response.setHeader("Access-Control-Allow-Credentials", "true");
// 允許跨域方法:全部允許為*,否則寫具體的方法,例如:GET, POST, PUT, DELETE, OPTIONS
response.setHeader("Access-Control-Allow-Methods", "*");
response.setHeader("Access-Control-Max-Age", "86400");
// 允許跨域請求頭
response.setHeader("Access-Control-Allow-Headers", "*");
// 如果是OPTIONS,不走后續(xù)操作,直接返回
if (HttpMethod.OPTIONS.toString().equals(request.getMethod())) {
response.setStatus(HttpServletResponse.SC_OK);
return false;
}
return true;
}
}- Access-Control-Allow-Origin: 允許跨域源,全部允許為*,否則寫允許請求的源ip。重點:通常會使用白名單機制,只有request.getHeader(“Origin”) 在白名單中才允許跨域。
- Access-Control-Allow-Methods:允許跨域方法:全部允許為*,否則寫具體的方法,例如:GET, POST, PUT, DELETE, OPTIONS
注冊攔截器,并指定攔截規(guī)則
在原有的InterceptorConfig 中增加注入CorsInterceptor ,并將【跨域攔截器】 放在 【身份認(rèn)證攔截器】之前,如下:
@Configuration
public class InterceptorConfig implements WebMvcConfigurer {
@Autowired
private CorsInterceptor corsInterceptor;
@Autowired
private AuthInterceptor authInterceptor;
@Override
public void addInterceptors(InterceptorRegistry registry) {
// 新增的跨域攔截器
registry.addInterceptor(corsInterceptor)
.addPathPatterns("/**");
// 身份認(rèn)證攔截器
registry.addInterceptor(authInterceptor)
.addPathPatterns("/**")
// 排除的請求路徑
.excludePathPatterns("/auth/login");
}
}去除原來的 @CrossOrigin注解,然后驗證,同樣,完全支持POST和OPTIONS方法!

三、過濾器方案
新增 CorsFilter 實現(xiàn)Filter接口
public class CorsFilter implements Filter {
@Override
public void init(FilterConfig filterConfig) throws ServletException {
}
@Override
public void doFilter(ServletRequest req, ServletResponse res, FilterChain chain)
throws IOException, ServletException {
HttpServletResponse response = (HttpServletResponse) res;
HttpServletRequest request = (HttpServletRequest) req;
if (StringUtils.isEmpty(request.getHeader("Origin"))) {
response.setHeader("Access-Control-Allow-Origin", "*");
} else {
response.setHeader("Access-Control-Allow-Origin", request.getHeader("Origin"));
}
response.setHeader("Access-Control-Allow-Credentials", "true");
response.setHeader("Access-Control-Allow-Methods", "*");
response.setHeader("Access-Control-Max-Age", "86400");
response.setHeader("Access-Control-Allow-Headers", "*");
// 如果是OPTIONS,不走后續(xù)操作,直接返回
if (HttpMethod.OPTIONS.toString().equals(request.getMethod())) {
response.setStatus(HttpServletResponse.SC_OK);
return;
}
chain.doFilter(request, response);
}
@Override
public void destroy() {
}
}新增配置類CorsConfig 注入 CorsFilter
@Configuration
public class CorsConfig {
@Bean
public CorsFilter corsFilter() {
return new CorsFilter();
}
}注意:測試時先注釋上面的 跨域攔截器,然后驗證,完全支持POST和OPTIONS方法!
總結(jié)
今天我使用了兩種方案解決跨域:攔截器和過濾器方案,都是超實用的方案。另外,給你留個作業(yè)吧,都做成配置擴展,而不是hard code,例如上面說到的白名單機制,實際就是配置一堆ip,其它也是同理,你可以把你的配置擴展寫在留言區(qū)里,或者遇到的問題在留言區(qū)和我一起討論,感謝你的閱讀,也歡迎你把這篇文章分享給更多的朋友一起閱讀。
以上就是SpringBoot解決跨域的超實用方案分享的詳細(xì)內(nèi)容,更多關(guān)于SpringBoot解決跨域的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Spring?Cloud?Eureka高可用配置(踩坑記錄)
在進行Eureka高可用配置時,控制臺一直出現(xiàn)“......”的錯誤,但是在瀏覽器中輸入地址:peer1:8761 卻是可正常運行,這篇文章主要介紹了Spring?Cloud踩坑之Eureka高可用配置,需要的朋友可以參考下2023-08-08
Idea導(dǎo)入eureka源碼實現(xiàn)過程解析
這篇文章主要介紹了Idea導(dǎo)入eureka源碼實現(xiàn)過程解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2020-08-08
Java 實戰(zhàn)圖書管理系統(tǒng)的實現(xiàn)流程
讀萬卷書不如行萬里路,只學(xué)書上的理論是遠(yuǎn)遠(yuǎn)不夠的,只有在實戰(zhàn)中才能獲得能力的提升,本篇文章手把手帶你用java+SSM+jsp+mysql+maven實現(xiàn)一個圖書管理系統(tǒng),大家可以在過程中查缺補漏,提升水平2021-11-11
Java8新特性之深入解析日期和時間_動力節(jié)點Java學(xué)院整理
這篇文章主要介紹了Java8新特性之深入解析日期和時間_動力節(jié)點Java學(xué)院整理,需要的朋友可以參考下2017-06-06
Spring?Boot?+?Mybatis?Plus實現(xiàn)樹狀菜單的方法
這篇文章主要介紹了Spring?Boot?+?Mybatis?Plus實現(xiàn)樹狀菜單,包括實體類中添加子菜單列表和集合及構(gòu)建菜單樹的詳細(xì)代碼,代碼簡單易懂,需要的朋友可以參考下2021-12-12

