• Springboot配置WebMvcConfig解决Cors非同源访问跨域问题


    关于Cors跨域的问题,前端有代理和jsonp的常用方式解决这种非同源的访问拒绝策略,什么是同源?即域名一致端口一致但是端口下访问的接口api不同的两种或者几种的互相访问叫做同源访问,但是若是接口不一致或者域名不一致(这里泛指IP不一致),那么对应的就属于非同源访问,浏览器会拒绝发出请求,直接回复404,有时候我也见过恢复202的就是发出去了但是被后端的Mvc处理hander链给拒绝了。那么配置MVC是后端处理Cors问题的一种解决思路。


    之前学习过MVC的处理链路,从一次请求发过来到回复数据总共11次处理:

     请求发送到服务器端时是由我们的MVC进行处理的,而统一调配任务流程的则是我们的请求分发器,注意这里请求到处理器之后回去寻找处理器适配器(符合校验处理的请求才能被允许例如接口含有的合法api,以及跨域原则),之前我们的微信小程序开发过程中是没有考虑跨域问题的,原因是我们知道小程序的请求处理都是由微信后台进行分发处理的,也就是在微信的后台时就做了前端的跨域处理,大概是采用动态代理的方式解决了小程序的跨域。

    那么我们先看看MVC的配置接口 WebMvcConfigurer 的源代码:

    1. public interface WebMvcConfigurer {
    2. default void configurePathMatch(PathMatchConfigurer configurer) {
    3. }
    4. default void configureContentNegotiation(ContentNegotiationConfigurer configurer) {
    5. }
    6. default void configureAsyncSupport(AsyncSupportConfigurer configurer) {
    7. }
    8. default void configureDefaultServletHandling(DefaultServletHandlerConfigurer configurer) {
    9. }
    10. default void addFormatters(FormatterRegistry registry) {
    11. }
    12. default void addInterceptors(InterceptorRegistry registry) {
    13. }
    14. default void addResourceHandlers(ResourceHandlerRegistry registry) {
    15. }
    16. default void addCorsMappings(CorsRegistry registry) {
    17. }
    18. default void addViewControllers(ViewControllerRegistry registry) {
    19. }
    20. default void configureViewResolvers(ViewResolverRegistry registry) {
    21. }
    22. default void addArgumentResolvers(List resolvers) {
    23. }
    24. default void addReturnValueHandlers(List handlers) {
    25. }
    26. default void configureMessageConverters(List> converters) {
    27. }
    28. default void extendMessageConverters(List> converters) {
    29. }
    30. default void configureHandlerExceptionResolvers(List resolvers) {
    31. }
    32. default void extendHandlerExceptionResolvers(List resolvers) {
    33. }
    34. @Nullable
    35. default Validator getValidator() {
    36. return null;
    37. }
    38. @Nullable
    39. default MessageCodesResolver getMessageCodesResolver() {
    40. return null;
    41. }
    42. }

    它的内部是具备一些处理器解析器以及映射的添加与配置的方法的,那么我们要解决Cros跨域问题就是要考虑 addCorsMappings 配置Cros映射,所以我们点进去看看这注册Cros的 CorsRegistry 的源码:

    1. public class CorsRegistry {
    2. private final List registrations = new ArrayList();
    3. public CorsRegistry() {
    4. }
    5. public CorsRegistration addMapping(String pathPattern) {
    6. CorsRegistration registration = new CorsRegistration(pathPattern);
    7. this.registrations.add(registration);
    8. return registration;
    9. }
    10. protected Map getCorsConfigurations() {
    11. Map configs = CollectionUtils.newLinkedHashMap(this.registrations.size());
    12. Iterator var2 = this.registrations.iterator();
    13. while(var2.hasNext()) {
    14. CorsRegistration registration = (CorsRegistration)var2.next();
    15. configs.put(registration.getPathPattern(), registration.getCorsConfiguration());
    16. }
    17. return configs;
    18. }
    19. }

    从上述代码中不难发现,内部有一个不可改变的 CorsRegistration  数组链表,以及增加映射的方法,主要还是看看它具备的元素 CorsRegistration 含有什么配置项:

    1. public class CorsRegistration {
    2. private final String pathPattern;
    3. private CorsConfiguration config;
    4. public CorsRegistration(String pathPattern) {
    5. this.pathPattern = pathPattern;
    6. this.config = (new CorsConfiguration()).applyPermitDefaultValues();
    7. }
    8. public CorsRegistration allowedOrigins(String... origins) {
    9. this.config.setAllowedOrigins(Arrays.asList(origins));
    10. return this;
    11. }
    12. public CorsRegistration allowedOriginPatterns(String... patterns) {
    13. this.config.setAllowedOriginPatterns(Arrays.asList(patterns));
    14. return this;
    15. }
    16. public CorsRegistration allowedMethods(String... methods) {
    17. this.config.setAllowedMethods(Arrays.asList(methods));
    18. return this;
    19. }
    20. public CorsRegistration allowedHeaders(String... headers) {
    21. this.config.setAllowedHeaders(Arrays.asList(headers));
    22. return this;
    23. }
    24. public CorsRegistration exposedHeaders(String... headers) {
    25. this.config.setExposedHeaders(Arrays.asList(headers));
    26. return this;
    27. }
    28. public CorsRegistration allowCredentials(boolean allowCredentials) {
    29. this.config.setAllowCredentials(allowCredentials);
    30. return this;
    31. }
    32. public CorsRegistration maxAge(long maxAge) {
    33. this.config.setMaxAge(maxAge);
    34. return this;
    35. }
    36. public CorsRegistration combine(CorsConfiguration other) {
    37. this.config = this.config.combine(other);
    38. return this;
    39. }
    40. protected String getPathPattern() {
    41. return this.pathPattern;
    42. }
    43. protected CorsConfiguration getCorsConfiguration() {
    44. return this.config;
    45. }
    46. }

    我们可以发现内部是具备允许放行:请求头,请求路径,请求方法,请求源策略的方法的,所以我们在这里的 重写addCorsMappings方法配置一个 CorsRegistry 添加相应的路径方法与请求策略放行不就可以解决跨域的问题了?


    我们写一个WebMvcConfig配置类实现刚刚研究的WebMvcConfigurer接口重写addCrosMappings配置CrosRegistry即可(或者在api与Controller控制类上打上@CrossOrigin注解也可以解决问题(注解默认放行所有来源的请求)):

    1. /**
    2. * 配置前端跨域访问请求
    3. */
    4. @Configuration
    5. public class WbMvcConfig implements WebMvcConfigurer {
    6. @Override
    7. public void addCorsMappings(CorsRegistry registry) {
    8. registry.addMapping("/**")
    9. .allowedHeaders("Content-Type","X-Request-With","Access-Control-Request-Method","Access-Control-Request-Headers","token")
    10. .allowedMethods("*")
    11. .allowedOriginPatterns("*")
    12. /*注意当这个配置为真是我们不能将允许源设置为*而是将源路径设置为*即可*/
    13. .allowCredentials(true);
    14. }
    15. @Bean
    16. public FormContentFilter httpPutFormContentFilter(){
    17. return new FormContentFilter();
    18. }
    19. }

    我们利用axios写一个简单的请求发送按钮:

    1. <input type="button" value="get" class="get">
    2. <script>
    3. document.querySelector(".get").onclick = function () {
    4. // 跨域一般是是后端解决的事情
    5. axios.get("http://127.0.0.1:8080/all").then(
    6. function (response) {
    7. console.log(response)
    8. }
    9. )
    10. }
    11. script>

    再用SpringBoot写一个简单的controller的api:

    1. @RestController
    2. public class testController {
    3. @Autowired
    4. private ProductServiceImpl productService;
    5. @GetMapping("/all")
    6. @ResponseBody
    7. public List all() {
    8. Page page = productService.page(1L);
    9. List productList = new LinkedList<>();
    10. productList.add(page.getRecords().iterator().next());
    11. return productList;
    12. }
    13. }

    这里我们在浏览器打开5050端口下的这个html文件就可以点击按钮访问接口了:

     这里可以看到请求访问数据成功了!

  • 相关阅读:
    使用 Docker Compose V2 快速搭建日志分析平台 ELK (Elasticsearch、Logstash 和 Kibana)
    OpenGLES:3D立方体纹理贴图
    【信号与系统】奇异信号的性质
    Docker:自定义镜像
    [暑假]操作系统概述笔记[学习方法篇]
    电商API接口解析及操作案例
    文件部署到服务器
    Vue3中全局配置 axios 的两种方式
    Java中,对象一定在堆中分配吗?
    Springboot项目集成Swagger3.0
  • 原文地址:https://blog.csdn.net/m0_59588838/article/details/127802639