Java實(shí)現(xiàn)CORS跨域請(qǐng)求的實(shí)現(xiàn)方法
問題
使用前后端分離模式開發(fā)項(xiàng)目時(shí),往往會(huì)遇到這樣一個(gè)問題 -- 無法跨域獲取服務(wù)端數(shù)據(jù)
這是由于瀏覽器的同源策略導(dǎo)致的,目的是為了安全。在前后端分離開發(fā)模式備受青睞的今天,前端和后臺(tái)項(xiàng)目往往會(huì)在不同的環(huán)境下進(jìn)行開發(fā),這時(shí)就會(huì)出現(xiàn)跨域請(qǐng)求數(shù)據(jù)的需求,目前的解決方案主要有以下幾種:
JSONP、iframe、代理模式、CORS等等
前面幾種方式在這里不講,網(wǎng)上有很多資料。在這里我主要分享一下CORS這種解決方式,CORS即“跨域資源共享”,它允許瀏覽器向跨源服務(wù)器,發(fā)出XMLHttpRequest請(qǐng)求,從而克服了AJAX只能同源使用的限制。
使用 CORS 跨域的時(shí)候和普通的 ajax 過程是一樣的,只是瀏覽器在發(fā)現(xiàn)這是一個(gè)跨域請(qǐng)求的時(shí)候會(huì)自動(dòng)幫我們處理一些事情,所以說只要服務(wù)端提供支持,前端是不需要做額外的事情的。
實(shí)現(xiàn)
實(shí)現(xiàn)的大概思路是這樣的,首先使用過濾器獲取請(qǐng)求對(duì)象request的信息,比如Origin 字段(表示請(qǐng)求來自哪個(gè)源,包括協(xié)議、域名、端口),通過預(yù)先配置的參數(shù)判斷請(qǐng)求是否合法,然后設(shè)置響應(yīng)對(duì)象response的頭信息,實(shí)現(xiàn)跨域資源請(qǐng)求。在介紹實(shí)現(xiàn)方式之前我們先來了解一下會(huì)用到的響應(yīng)頭信息。
響應(yīng)頭
Access-Control-Allow-Methods
用來列出瀏覽器的CORS請(qǐng)求允許使用的HTTP方法,如:GET、POST、PUT、DELETE、OPTIONS
Access-Control-Allow-Credentials
表示是否支持跨域Cookie
Access-Control-Allow-Headers
逗號(hào)分隔的字符串,表示服務(wù)器支持的所有頭信息字段,如Content-Type以及自定義的字段
Access-Control-Expose-Headers
與“Access-Control-Allow-Headers”相反,表示不支持的頭信息字段
Access-Control-Allow-Origin
允許跨域的請(qǐng)求源信息,包括協(xié)議、域名、端口,為*表示允許所有請(qǐng)求來源,并且只能設(shè)置一個(gè)請(qǐng)求源
下面介紹一下Java后臺(tái)如何實(shí)現(xiàn)這種方式。
代碼
由于最近在使用spring-boot,所以接下來以spring-boot為基礎(chǔ)來實(shí)現(xiàn)。
首先創(chuàng)建一個(gè)CorsFilter過濾器,代碼如下:
...
@WebFilter(filterName = "corsFilter", urlPatterns = "/*",
initParams = {@WebInitParam(name = "allowOrigin", value = "*"),
@WebInitParam(name = "allowMethods", value = "GET,POST,PUT,DELETE,OPTIONS"),
@WebInitParam(name = "allowCredentials", value = "true"),
@WebInitParam(name = "allowHeaders", value = "Content-Type,X-Token")})
public class CorsFilter implements Filter {
private String allowOrigin;
private String allowMethods;
private String allowCredentials;
private String allowHeaders;
private String exposeHeaders;
@Override
public void init(FilterConfig filterConfig) throws ServletException {
allowOrigin = filterConfig.getInitParameter("allowOrigin");
allowMethods = filterConfig.getInitParameter("allowMethods");
allowCredentials = filterConfig.getInitParameter("allowCredentials");
allowHeaders = filterConfig.getInitParameter("allowHeaders");
exposeHeaders = filterConfig.getInitParameter("exposeHeaders");
}
@Override
public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {
HttpServletRequest request = (HttpServletRequest) servletRequest;
HttpServletResponse response = (HttpServletResponse) servletResponse;
if (!StringUtils.isEmpty(allowOrigin)) {
if(allowOrigin.equals("*")){
response.setHeader("Access-Control-Allow-Origin", allowOrigin);
}else{
List<String> allowOriginList = Arrays.asList(allowOrigin.split(","));
if (allowOriginList != null && allowOriginList.size() > 0) {
String currentOrigin = request.getHeader("Origin");
if (allowOriginList.contains(currentOrigin)) {
response.setHeader("Access-Control-Allow-Origin", currentOrigin);
}
}
}
}
if (!StringUtils.isEmpty(allowMethods)) {
response.setHeader("Access-Control-Allow-Methods", allowMethods);
}
if (!StringUtils.isEmpty(allowCredentials)) {
response.setHeader("Access-Control-Allow-Credentials", allowCredentials);
}
if (!StringUtils.isEmpty(allowHeaders)) {
response.setHeader("Access-Control-Allow-Headers", allowHeaders);
}
if (!StringUtils.isEmpty(exposeHeaders)) {
response.setHeader("Access-Control-Expose-Headers", exposeHeaders);
}
filterChain.doFilter(servletRequest, servletResponse);
}
@Override
public void destroy() {
}
}
大功告成,現(xiàn)在前端就可以跨域獲取后臺(tái)的數(shù)據(jù)了,比其它方式容易得多,代碼就不解釋了,簡(jiǎn)單易懂,使用其它后臺(tái)開發(fā)方式也一樣,最終目的就是判斷請(qǐng)求,設(shè)置響應(yīng)頭,前端什么事都不用做。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
ShardingSphere結(jié)合MySQL實(shí)現(xiàn)分庫分表的項(xiàng)目實(shí)踐
在實(shí)際開發(fā)中,如果表的數(shù)據(jù)過大我們需要把一張表拆分成多張表,本文主要介紹了使用ShardingSphere實(shí)現(xiàn)MySQL分庫分表,具有一定的參考價(jià)值,感興趣的可以了解一下2024-03-03
完美解決springboot項(xiàng)目出現(xiàn)”java: 錯(cuò)誤: 無效的源發(fā)行版:17“問題(圖文詳解)
這篇文章主要介紹了完美解決springboot項(xiàng)目出現(xiàn)”java: 錯(cuò)誤: 無效的源發(fā)行版:17“問題,本文通過圖文并茂的形式給大家介紹的非常詳細(xì),需要的朋友可以參考下2023-04-04
java實(shí)現(xiàn)檢測(cè)是否字符串中包含中文
本文給大家分享了2個(gè)使用java檢測(cè)字符串中是否包含中文的代碼,都非常的實(shí)用,最后附上了各種字符的unicode編碼的范圍,方便我們以后使用正則進(jìn)行匹配檢測(cè)。2015-10-10
@WebFilter在SpringBoot無效的原因分析和解決方案
使用Ruoyi的demo部署成功后,發(fā)現(xiàn)js、css等靜態(tài)文件都進(jìn)入了過濾器,但是發(fā)現(xiàn)靜態(tài)文件沒有使用瀏覽器緩存,新建BrowserCacheFilter.java并增加@WebFilter處理,應(yīng)用自動(dòng)重啟后發(fā)現(xiàn)@WebFilter無效,所以本文給大家介紹了@WebFilter在SpringBoot無效的原因分析和解決方案2024-03-03

