Spring的跨域的幾個(gè)方案
1.@CrossOrigin
@CrossOrigin可以添加到方法上,也可以添加到Controller上
AbstractHandlerMethodMapping的內(nèi)部類MappingRegistry的register:
public void register(T mapping, Object handler, Method method) {
? ?// Assert that the handler method is not a suspending one.
? ?if (KotlinDetector.isKotlinType(method.getDeclaringClass())) {
? ? ? Class<?>[] parameterTypes = method.getParameterTypes();
? ? ? if ((parameterTypes.length > 0) && "kotlin.coroutines.Continuation".equals(parameterTypes[parameterTypes.length - 1].getName())) {
? ? ? ? ?throw new IllegalStateException("Unsupported suspending handler method detected: " + method);
? ? ? }
? ?}
? ?this.readWriteLock.writeLock().lock();
? ?try {
? ? ? HandlerMethod handlerMethod = createHandlerMethod(handler, method);
? ? ? validateMethodMapping(handlerMethod, mapping);
? ? ? this.mappingLookup.put(mapping, handlerMethod);
? ? ? List<String> directUrls = getDirectUrls(mapping);
? ? ? for (String url : directUrls) {
? ? ? ? ?this.urlLookup.add(url, mapping);
? ? ? }
? ? ? String name = null;
? ? ? if (getNamingStrategy() != null) {
? ? ? ? ?name = getNamingStrategy().getName(handlerMethod, mapping);
? ? ? ? ?addMappingName(name, handlerMethod);
? ? ? }
? ? ? CorsConfiguration corsConfig = initCorsConfiguration(handler, method, mapping);
? ? ? if (corsConfig != null) {
? ? ? ? ?this.corsLookup.put(handlerMethod, corsConfig);
? ? ? }
? ? ? this.registry.put(mapping, new MappingRegistration<>(mapping, handlerMethod, directUrls, name));
? ?}
? ?finally {
? ? ? this.readWriteLock.writeLock().unlock();
? ?}
}@CrossOrigin注解在AbstractHandlerMethodMapping的內(nèi)部類MappingRegistry的register方法中完成解析,@CrossOrigin注解中的內(nèi)容會(huì)被解析成一個(gè)配置對(duì)象CorsConfiguration- 將
@CrossOrigin所標(biāo)記的請(qǐng)求方法對(duì)象HandlerMethod和CorsConfiguration一一對(duì)應(yīng)存入corsLookup的map集合中 - 當(dāng)請(qǐng)求到達(dá)
DispatcherServlet的doDispatch方法之后,調(diào)用AbstractHandlerMapping的getHandler方法獲取執(zhí)行鏈HandlerExecutionChain時(shí),會(huì)從map中獲取CorsConfiguration對(duì)象 - 根據(jù)獲取到的
CorsConfiguration對(duì)象構(gòu)建一個(gè)CorsInterceptor攔截器 - 在
CorsInterceptor攔截器中觸發(fā)對(duì)CorsProcessor的processRequest方法調(diào)用,跨域請(qǐng)求的校驗(yàn)工作將在該方法中完成。
2.addCorsMappings
@CrossOrigin是添加在不同的Controller中 全局配置
@Configuration
public class WebMvcConfig implements WebMvcConfigurer {
? ? @Override
? ? public void addCorsMappings(CorsRegistry registry) {
? ? ? ? registry.addMapping("/**")
? ? ? ? ? ? ? ? .allowedMethods("*")
? ? ? ? ? ? ? ? .allowedOrigins("*")
? ? ? ? ? ? ? ? .allowedHeaders("*")
? ? ? ? ? ? ? ? .allowCredentials(false)
? ? ? ? ? ? ? ? .exposedHeaders("")
? ? ? ? ? ? ? ? .maxAge(3600);
? ? }
}全局配置和@CrossOrigin注解相同,都是在CorsInterceptor攔截器中觸發(fā)對(duì)CorsProcessor的processRequest方法調(diào)用,最終在該方法中完成跨域請(qǐng)求的校驗(yàn)工作
registry.addMapping(“/**”)方法中配置了一個(gè)CorsRegistration對(duì)象,該對(duì)象中包含了一個(gè)路徑攔截規(guī)則,同時(shí)CorsRegistration還包含了一個(gè)CorsConfiguration配置對(duì)象,該對(duì)象用來(lái)保存這里跨域相關(guān)的配置。- 在
WebMvcConfigurationSupport的requestMappingHandlerMapping方法中觸發(fā)了addCorsMappings方法執(zhí)行,將獲取到的CorsRegistration對(duì)象重新組裝成一個(gè)UrlBasedCorsConfigurationSource對(duì)象,該對(duì)象保存了攔截規(guī)則和CorsConfiguration對(duì)象的映射關(guān)系。 - 將新建的
UrlBasedCorsConfigurationSource對(duì)象賦值給AbstractHandlerMapping的corsConfigurationSource屬性 - 當(dāng)請(qǐng)求到達(dá)時(shí)的處理方法和
@CrossOrigin注解處理流程一樣,在AbstractHandlerMapping的getHandler方法處理,從corsConfigurationSource中獲取CorsConfiguration配置對(duì)象,而@CrossOrigin從map中獲取CorsConfiguration對(duì)象。如果兩處都可以獲取到CorsConfiguration對(duì)象,則獲取到的對(duì)象屬性值進(jìn)行合并。 - 根據(jù)獲取到的
CorsConfiguration對(duì)象構(gòu)造CorsInterceptor攔截器 - 在
CorsInterceptor攔截器中觸發(fā)對(duì)CorsProcessor的processRequest方法調(diào)用,跨域請(qǐng)求的校驗(yàn)工作將在該方法中完成。
這里的跨域校驗(yàn)是通過(guò)DispatcherServlet中的方法觸發(fā)的,DispatcherServlet在Filter之后執(zhí)行
3.CorsFIlter
@Configuration
public class WebMvcConfig {
? ? @Bean
? ? FilterRegistrationBean<CorsFilter> corsFilter() {
? ? ? ? FilterRegistrationBean<CorsFilter> registrationBean = new FilterRegistrationBean<>();
? ? ? ? CorsConfiguration corsConfiguration = new CorsConfiguration();
? ? ? ? corsConfiguration.setAllowedHeaders(Arrays.asList("*"));
? ? ? ? corsConfiguration.setAllowedMethods(Arrays.asList("*"));
? ? ? ? corsConfiguration.setAllowedOrigins(Arrays.asList("http://localhost:8081"));
? ? ? ? corsConfiguration.setMaxAge(3600L);
? ? ? ? UrlBasedCorsConfigurationSource source = new UrlBasedCorsConfigurationSource();
? ? ? ? source.registerCorsConfiguration("/**", corsConfiguration);
? ? ? ? registrationBean.setFilter(new CorsFilter(source));
? ? ? ? registrationBean.setOrder(-1);
? ? ? ? return registrationBean;
? ? }
}- 手動(dòng)創(chuàng)建
CorsConfiguration對(duì)象 - 創(chuàng)建
UrlBasedCorsConfigurationSource對(duì)象,將過(guò)濾器的攔截規(guī)則和CorsConfiguration對(duì)象之間的映射關(guān)系由UrlBasedCorsConfigurationSource中的corsConfiguration變量保存起來(lái)。 - 最后創(chuàng)建
CorsFilter設(shè)置優(yōu)先級(jí)
CorsFilter的doFilterInternal方法:
protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain filterChain) throws ServletException, IOException {
? ? CorsConfiguration corsConfiguration = this.configSource.getCorsConfiguration(request);
? ? boolean isValid = this.processor.processRequest(corsConfiguration, request, response);
? ? if (isValid && !CorsUtils.isPreFlightRequest(request)) {
? ? ? ? filterChain.doFilter(request, response);
? ? }
}觸發(fā)對(duì)CorsProcessor的processRequest方法調(diào)用,跨域請(qǐng)求的校驗(yàn)工作將在該方法中完成
到此這篇關(guān)于Spring的跨域的幾個(gè)方案的文章就介紹到這了,更多相關(guān)Spring的跨域方案內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
java實(shí)現(xiàn)的連接數(shù)據(jù)庫(kù)及模糊查詢功能示例
這篇文章主要介紹了java實(shí)現(xiàn)的連接數(shù)據(jù)庫(kù)及模糊查詢功能,結(jié)合實(shí)例形式分析了java基于jdbc連接數(shù)據(jù)庫(kù)及使用LIKE語(yǔ)句實(shí)現(xiàn)模糊查詢功能的相關(guān)操作技巧,需要的朋友可以參考下2017-12-12
Java并發(fā)Map面試線程安全數(shù)據(jù)結(jié)構(gòu)全面分析
本文將探討如何在Java中有效地應(yīng)對(duì)這些挑戰(zhàn),介紹一種強(qiáng)大的工具并發(fā)Map,它能夠幫助您管理多線程環(huán)境下的共享數(shù)據(jù),確保數(shù)據(jù)的一致性和高性能,深入了解Java中的并發(fā)Map實(shí)現(xiàn),包括ConcurrentHashMap和ConcurrentSkipListMap,及相關(guān)知識(shí)點(diǎn)2023-09-09
關(guān)于jpa中無(wú)法刪除onetomany中many問(wèn)題的解決
這篇文章主要介紹了關(guān)于jpa中無(wú)法刪除onetomany中many問(wèn)題的解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-12-12
Java實(shí)現(xiàn)爬取往期所有雙色球開獎(jiǎng)結(jié)果功能示例
這篇文章主要介紹了Java實(shí)現(xiàn)爬取往期所有雙色球開獎(jiǎng)結(jié)果功能,涉及Java網(wǎng)頁(yè)抓取、正則替換、文件讀寫等相關(guān)操作技巧,需要的朋友可以參考下2018-07-07
詳解SpringMVC注解版前臺(tái)向后臺(tái)傳值的兩種方式
本篇文章主要介紹了詳解SpringMVC注解版前臺(tái)向后臺(tái)傳值的兩種方式,具有一定的參考價(jià)值,有興趣的可以了解一下。2017-04-04

