美文网首页
springboot 2.0设置允许跨域

springboot 2.0设置允许跨域

作者: blog星洛 | 来源:发表于2020-01-09 09:14 被阅读0次

​ 目前项目上采用架构是springboot 2.0版本,前后端分离项目。

​ 由于前后端项目不在同一个服务器上,所以前端请求后端数据时,会出现跨域问题。这时候就需要设置允许跨域。

一般有以下几种方式。

  1. 在controller类上添加@CrossOrigin注解

    @Controller
    @CrossOrigin
    public class WebController {
    
    }
    

    但是这种方式需要在每个controller上添加,太麻烦。不采用。

  1. 添加全局跨域配置类

    继承WebMvcConfigurationSupport类。实现里面的方法。

    addCorsMappings方法详解:

    allowedOrigins:允许所有域名访问。

    allowCredentials:允许携带缓存。

    allowedMethods:允许通过的方法。

    maxAge:设置预请求的生效时间,发过一次后,一定时间内不在发送。

    allowedHeaders:允许获取所有的头信息。

    其他方法和跨域无关。

package com.standard.config;

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.web.servlet.config.annotation.*;

@Configuration
public class CorsConfig extends WebMvcConfigurationSupport {

    @Override
    public void addCorsMappings(CorsRegistry registry) {
        registry.addMapping("/**")
                .allowedOrigins("*")
                .allowCredentials(true)
                .allowedMethods("GET", "POST", "PUT", "DELETE", "OPTIONS")
                .maxAge(3600)
                .allowedHeaders("*");
    }

    @Override
    public void addResourceHandlers(ResourceHandlerRegistry registry) {

        registry.addResourceHandler("swagger-ui.html")
                .addResourceLocations("classpath:/META-INF/resources/");

        registry.addResourceHandler("/webjars/**")
                .addResourceLocations("classpath:/META-INF/resources/webjars/");
    }

    /**
     * 配置的拦截器
     * @return
     */
    @Bean
    public SecurityInterceptor getSecurityInterceptor() {
        return new SecurityInterceptor();
    }

    /**
     * 配置的拦截器
     * @return
     */
    @Bean
    public AdminInterceptor getAdminInterceptor() {
        return new AdminInterceptor();
    }

    /**
     * 拦截的url
     * @param registry
     */
    @Override
    public void addInterceptors(InterceptorRegistry registry) {
        InterceptorRegistration addInterceptor = registry.addInterceptor(getSecurityInterceptor());

//        InterceptorRegistration registration = registry.addInterceptor(getAdminInterceptor());
//        registration.addPathPatterns("/sysconfig/**");                      //所有路径都被拦截
//        registration.excludePathPatterns(
//                "user/**",          //添加不拦截路径
//                "user/login",            //登录
//                "/**/*.html",            //html静态资源
//                "/**/*.js",              //js静态资源
//                "/**/*.css",             //css静态资源
//                "/**/*.woff",
//                "/**/*.ttf"
//        );
        // 排除配置--对下面的不进行拦截
        //  addInterceptor.excludePathPatterns("/user/**");
        // 拦截配置
        addInterceptor.addPathPatterns("/homepage/**");
    }
}

  1. 通过过滤器实现

​ 实现原理第二种没区别,实现方式不同而已。

package com.xdx97.backstage.filter;
 
//判断是否登陆的 filter
 
import javax.servlet.*;
import javax.servlet.annotation.WebFilter;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.util.Arrays;
import java.util.Collections;
import java.util.HashSet;
import java.util.Set;
 
// /* 表示全部拦截
@WebFilter(filterName = "loginFilter",urlPatterns = "/*")
public class LoginFilter implements Filter {
    //这里面 填写不需要 被拦截的地址
    private static final Set<String> ALLOWED_PATHS = Collections.unmodifiableSet(
            new HashSet<String>( Arrays.asList("/login","/isLogin","/findCategory") )
    );
 
    //初始化调用的方法
    //当服务器 被启动的时候,调用
    public void init(FilterConfig filterConfig) throws ServletException { }
 
    //拦截的方法
    public void doFilter(ServletRequest req, ServletResponse res, FilterChain filterChain) throws IOException, ServletException {
 
 
        HttpServletRequest request = (HttpServletRequest) req;
        HttpServletResponse response = (HttpServletResponse) res;
 
        //解决跨域的问题
        response.setHeader("Access-Control-Allow-Origin","http://localhost:3030");
        response.setHeader("Access-Control-Allow-Credentials","true");
        response.setHeader("Access-Control-Allow-Headers", "Content-Type,Content-Length, Authorization, Accept,X-                                                                Requested-With,X-App-Id, X-Token");
        response.setHeader("Access-Control-Allow-Methods","PUT,POST,GET,DELETE,OPTIONS");
        response.setHeader("Access-Control-Max-Age", "3600");
 
 
        String path = request.getRequestURI().substring(request.getContextPath().length()).replaceAll("[/]+$","");
        boolean allowePath =  ALLOWED_PATHS.contains(path);
 
        if (!allowePath) {  //需要拦截的方法
            Object aa = request.getSession().getAttribute("user");
            if (aa != null) {
                filterChain.doFilter(request,response);
        }
            else {
                response.getWriter().write("noLogin");
            }
        } else {  //不需要被拦截的方法
            //直接放行
            filterChain.doFilter(request,response);
        }
 
 
 
    }
 
    //销毁时候调用的方法
    public void destroy() { }
}

重要会遇到问题

一开始感觉没啥问题,后来实际上出的问题导致浪费了很长时间在上面。

问题就是前端对于跨域请求会发送预请求

后端虽然实现了跨域支持,但是后端会对url进行一次拦截。用来校验是否登录。

然后前端发送预请求时,是不会携带登录信息的。这就导致前端一直访问不了后台页面。解决办法就是把预请求直接放行。

if (RequestMethod.OPTIONS.name().equals(request.getMethod())) {
    return super.preHandle(request, response, handler);
 }

以下shiro中放开过滤做参考:

Shiro中放开预请求的拦截

Shiro中放开预请求的拦截2

相关文章

网友评论

      本文标题:springboot 2.0设置允许跨域

      本文链接:https://www.haomeiwen.com/subject/wrrsactx.html