Spring Boot-CORS
什么是跨域?
定义:浏览器从一个域名的网页取请求另一个域名下的东西。通俗点说,浏览器直接从A域访问B域中的资源是不被允许的,如果想要访问,就需要进行一步操作,这操作就叫“跨域”。例如,你从百度的页面,点击一个按钮,请求了新浪的一个接口,这就进行了跨域。不单单只有域名不同就是跨域,域名、端口、协议其一不同就是不同的域,请求资源需要跨域。
为什么要跨域?
为什么需要跨域,而不直接访问其他域下的资源呢?这是浏览器的限制,专业点说叫浏览器同源策略限制。主要是为了安全考虑。现在的安全框架,一般请求的时候header中不是都存个token嘛,你要是用这个token去正常访问A域下的东西是没问题的,然后又去访问了B域,结果阴差阳错的还带着这个token,那么B域,或者说B网站是不是就可以拿着你的token去A域下做点什么呢,这就相当危险了。所以浏览器加上了所谓的浏览器同源策略限制。但是为了我们真的需要从A域下访问B的资源(正常访问),就需要用到跨域,跨越这个限制了。
SpringBoot解决跨域问题
SpringBoot可以基于Cors解决跨域问题,Cors是一种机制,告诉我们的后台,哪边(origin )来的请求可以访问服务器的数据。
全局配置
配置实例如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 @Configuration public class CorsConfig { @Bean public WebMvcConfigurer corsConfigurer () { return new WebMvcConfigurer(){ @Override public void addCorsMappings (CorsRegistry registry) { registry.addMapping("/**" ) .allowedOrigins("*" ) .allowedMethods("POST" , "GET" , "PUT" , "OPTIONS" , "DELETE" ) .allowCredentials(true ) .maxAge(3600 ); } }; } }
首先实现了WebMvcConfigurer
接口,WebMvcConfigurer
这个接口十分强大,里面还有很多可用的方法,在SpringBoot2.0
里面可以解决WebMvcConfigurerAdapter
曾经的部分任务。其中一个方法就是addCorsMappings()
,是专门为开发人员解决跨域而诞生的接口。其中构造参数为CorsRegistry
。
CorsRegistry的源码
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 public class CorsRegistry { private final List<CorsRegistration> registrations = new ArrayList<>(); public CorsRegistration addMapping (String pathPattern) { CorsRegistration registration = new CorsRegistration(pathPattern); this .registrations.add(registration); return registration; } protected Map<String, CorsConfiguration> getCorsConfigurations () { Map<String, CorsConfiguration> configs = new LinkedHashMap<>(this .registrations.size()); for (CorsRegistration registration : this .registrations) { configs.put(registration.getPathPattern(), registration.getCorsConfiguration()); } return configs; } }
可以看出CorsRegistry
有个属性registrations
,按道理可以根据不同的项目路径进行定制访问行为,但是我们示例直接将pathPattern
设置为/**
,也就是说已覆盖项目所有路径,只需要创建一个CorsRegistration就好。getCorsConfigurations()
,这个方法是获取所有CorsConfiguration
的Map集合,key值为传入路径pathPattern
。
回到示例代码CorsConfig中,registry
对象addMapping()
增加完传入路径pathPattern
之后,return了一个CorsRegistration
对象,是进行更多的配置,看一下CorsRegistration
的代码,看看我们能配些什么?
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 public class CorsRegistration { private final String pathPattern; private final CorsConfiguration config; public CorsRegistration (String pathPattern) { this .pathPattern = pathPattern; this .config = new CorsConfiguration().applyPermitDefaultValues(); } public CorsRegistration allowedOrigins (String... origins) { this .config.setAllowedOrigins(Arrays.asList(origins)); return this ; } public CorsRegistration allowedMethods (String... methods) { this .config.setAllowedMethods(Arrays.asList(methods)); return this ; } public CorsRegistration allowedHeaders (String... headers) { this .config.setAllowedHeaders(Arrays.asList(headers)); return this ; } public CorsRegistration exposedHeaders (String... headers) { this .config.setExposedHeaders(Arrays.asList(headers)); return this ; } public CorsRegistration allowCredentials (boolean allowCredentials) { this .config.setAllowCredentials(allowCredentials); return this ; } public CorsRegistration maxAge (long maxAge) { this .config.setMaxAge(maxAge); return this ; } protected String getPathPattern () { return this .pathPattern; } protected CorsConfiguration getCorsConfiguration () { return this .config; } }
局部配置
在Controller上加入@CrossOrigin
注解
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 @Target ({ ElementType.METHOD, ElementType.TYPE })@Retention (RetentionPolicy.RUNTIME)@Documented public @interface CrossOrigin { @Deprecated String[] DEFAULT_ORIGINS = { "*" }; @Deprecated String[] DEFAULT_ALLOWED_HEADERS = { "*" }; @Deprecated boolean DEFAULT_ALLOW_CREDENTIALS = false ; @Deprecated long DEFAULT_MAX_AGE = 1800 @AliasFor ("origins" ) String[] value() default {}; @AliasFor ("value" ) String[] origins() default {}; String[] allowedHeaders() default {}; String[] exposedHeaders() default {}; RequestMethod[] methods() default {}; String allowCredentials () default "" ; long maxAge () default -1 ; }
这个注解可以作用于方法或者类上,实现局部跨域,你会发现除了设置路径(因为没必要了,都定位到局部了)其他的参数与全局类似。