Java 如何解決跨域問題
我們?cè)陂_發(fā)過程中經(jīng)常會(huì)遇到前后端分離而導(dǎo)致的跨域問題,導(dǎo)致無法獲取返回結(jié)果。跨域就像分離前端和后端的一道鴻溝,君在這邊,她在那邊,兩兩不能往來.
什么是跨域(CORS)跨域(CORS)是指不同域名之間相互訪問。跨域,指的是瀏覽器不能執(zhí)行其他網(wǎng)站的腳本,它是由瀏覽器的同源策略所造成的,是瀏覽器對(duì)于JavaScript所定義的安全限制策略。
什么情況會(huì)跨域 同一協(xié)議, 如http或https 同一IP地址, 如127.0.0.1 同一端口, 如8080以上三個(gè)條件中有一個(gè)條件不同就會(huì)產(chǎn)生跨域問題。
解決方案前端解決方案 使用JSONP方式實(shí)現(xiàn)跨域調(diào)用; 使用NodeJS服務(wù)器做為服務(wù)代理,前端發(fā)起請(qǐng)求到NodeJS服務(wù)器, NodeJS服務(wù)器代理轉(zhuǎn)發(fā)請(qǐng)求到后端服務(wù)器; 后端解決方案 nginx反向代理解決跨域 服務(wù)端設(shè)置Response Header(響應(yīng)頭部)的Access-Control-Allow-Origin 在需要跨域訪問的類和方法中設(shè)置允許跨域訪問(如Spring中使用@CrossOrigin注解); 繼承使用Spring Web的CorsFilter(適用于Spring MVC、Spring Boot) 實(shí)現(xiàn)WebMvcConfigurer接口(適用于Spring Boot)具體方式一、使用Filter方式進(jìn)行設(shè)置使用Filter過濾器來過濾服務(wù)請(qǐng)求,向請(qǐng)求端設(shè)置Response Header(響應(yīng)頭部)的Access-Control-Allow-Origin屬性聲明允許跨域訪問。
@WebFilterpublic class CorsFilter implements Filter { @Override public void doFilter(ServletRequest req, ServletResponse res, FilterChain chain) throws IOException, ServletException { HttpServletResponse response = (HttpServletResponse) res; response.setHeader('Access-Control-Allow-Origin', '*'); response.setHeader('Access-Control-Allow-Methods', '*'); response.setHeader('Access-Control-Max-Age', '3600'); response.setHeader('Access-Control-Allow-Headers', '*'); response.setHeader('Access-Control-Allow-Credentials', 'true'); chain.doFilter(req, res); } }二、繼承 HandlerInterceptorAdapter
@Componentpublic class CrossInterceptor extends HandlerInterceptorAdapter { @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { response.setHeader('Access-Control-Allow-Origin', '*'); response.setHeader('Access-Control-Allow-Methods', 'GET, POST, PUT, DELETE, OPTIONS'); response.setHeader('Access-Control-Max-Age', '3600'); response.setHeader('Access-Control-Allow-Headers', '*'); response.setHeader('Access-Control-Allow-Credentials', 'true'); return true; }}三、實(shí)現(xiàn) WebMvcConfigurer
@Configuration@SuppressWarnings('SpringJavaAutowiredFieldsWarningInspection')public class AppConfig implements WebMvcConfigurer { @Override public void addCorsMappings(CorsRegistry registry) { registry.addMapping('/**') // 攔截所有的請(qǐng)求.allowedOrigins('http://www.abc.com') // 可跨域的域名,可以為 *.allowCredentials(true).allowedMethods('*') // 允許跨域的方法,可以單獨(dú)配置.allowedHeaders('*'); // 允許跨域的請(qǐng)求頭,可以單獨(dú)配置 }}四、使用Nginx配置
location / { add_header Access-Control-Allow-Origin *; add_header Access-Control-Allow-Headers X-Requested-With; add_header Access-Control-Allow-Methods GET,POST,PUT,DELETE,OPTIONS; if ($request_method = ’OPTIONS’) { return 204; }}五、使用 @CrossOrgin 注解
如果只是想部分接口跨域,且不想使用配置來管理的話,可以使用這種方式
在Controller使用
@CrossOrigin@RestController@RequestMapping('/user')public class UserController {@GetMapping('/{id}')public User get(@PathVariable Long id) {}@DeleteMapping('/{id}')public void remove(@PathVariable Long id) {}}
在具體接口上使用
@RestController@RequestMapping('/user')public class UserController {@CrossOrigin@GetMapping('/{id}')public User get(@PathVariable Long id) {}@DeleteMapping('/{id}')public void remove(@PathVariable Long id) {}}Spring Cloud Gateway 跨域配置
spring: cloud: gateway: globalcors: cors-configurations: ’[/**]’: # 允許跨域的源(網(wǎng)站域名/ip),設(shè)置*為全部 # 允許跨域請(qǐng)求里的head字段,設(shè)置*為全部 # 允許跨域的method, 默認(rèn)為GET和OPTIONS,設(shè)置*為全部 allow-credentials: true allowed-origins: - 'http://xb.abc.com' - 'http://sf.xx.com' allowed-headers: '*' allowed-methods: - OPTIONS - GET - POST - DELETE - PUT - PATCH max-age: 3600
注意: 通過gateway 轉(zhuǎn)發(fā)的其他項(xiàng)目,不要進(jìn)行配置跨域配置
有時(shí)即使配置了也不會(huì)起作用,這時(shí)你可以根據(jù)瀏覽器控制的錯(cuò)誤輸出來查看問題,如果提示是 response 中 header 出現(xiàn)了重復(fù)的 Access-Control-* 請(qǐng)求頭,可以進(jìn)行如下操作
import java.util.ArrayList;import org.springframework.cloud.gateway.filter.GatewayFilterChain;import org.springframework.cloud.gateway.filter.GlobalFilter;import org.springframework.cloud.gateway.filter.NettyWriteResponseFilter;import org.springframework.core.Ordered;import org.springframework.http.HttpHeaders;import org.springframework.stereotype.Component;import org.springframework.web.server.ServerWebExchange;import reactor.core.publisher.Mono;@Component('corsResponseHeaderFilter')public class CorsResponseHeaderFilter implements GlobalFilter, Ordered { @Override public int getOrder() { // 指定此過濾器位于NettyWriteResponseFilter之后 // 即待處理完響應(yīng)體后接著處理響應(yīng)頭 return NettyWriteResponseFilter.WRITE_RESPONSE_FILTER_ORDER + 1; } @Override public Mono<Void> filter(ServerWebExchange exchange, GatewayFilterChain chain) { return chain.filter(exchange).then(Mono.defer(() -> { exchange.getResponse().getHeaders().entrySet().stream() .filter(kv -> (kv.getValue() != null && kv.getValue().size() > 1)) .filter(kv -> ( kv.getKey().equals(HttpHeaders.ACCESS_CONTROL_ALLOW_ORIGIN) || kv.getKey().equals(HttpHeaders.ACCESS_CONTROL_ALLOW_CREDENTIALS) || kv.getKey().equals(HttpHeaders.ACCESS_CONTROL_ALLOW_METHODS) || kv.getKey().equals(HttpHeaders.ACCESS_CONTROL_ALLOW_HEADERS) || kv.getKey().equals(HttpHeaders.ACCESS_CONTROL_MAX_AGE))) .forEach(kv -> { kv.setValue(new ArrayList<String>() {{ add(kv.getValue().get(0)); }}); }); return chain.filter(exchange); })); }}
以上就是Java 如何解決跨域問題的詳細(xì)內(nèi)容,更多關(guān)于Java 解決跨域問題的資料請(qǐng)關(guān)注好吧啦網(wǎng)其它相關(guān)文章!
相關(guān)文章:
1. asp讀取xml文件和記數(shù)2. IDEA中 Getter、Setter 注解不起作用的問題如何解決3. Android CountDownTimer案例總結(jié)4. 簡(jiǎn)體中文轉(zhuǎn)換為繁體中文的PHP函數(shù)5. Python 中如何使用 virtualenv 管理虛擬環(huán)境6. 多個(gè)SpringBoot項(xiàng)目采用redis實(shí)現(xiàn)Session共享功能7. python利用opencv實(shí)現(xiàn)顏色檢測(cè)8. CSS自定義滾動(dòng)條樣式案例詳解9. 每日六道java新手入門面試題,通往自由的道路第二天10. PHP實(shí)現(xiàn)基本留言板功能原理與步驟詳解
