后端实现跨域的主流方法
- springMvc 4.2 以上
@CrossOrigin
添加到对应需要跨域的接口类或方法上,不加任何参数默认是所有网站都可以访问! - 通过拦截器、过滤器实现跨域,springMvc 4.2 以下,重写拦截器和过滤器即可。
这里说一下为什么可以直接用注解完成过滤却还要重写拦截器:
鄙人技术有限,在用拦截器做完登陆验证,无法通过拦截器实现未被拦截器拦截接口的跨域功能,故另辟蹊径使用注解完成未被拦截接口的跨域功能。
拦截器通常实现的功能:
- 登录验证,判断用户是否登录。
- 判断用户是否有权限访问资源,如校验token
- 记录请求操作日志(用户ip,访问时间等),以便统计请求访问量。
- cookie、本地化、国际化、主题等。
- 监控请求处理时长等。
@CrossOrigin
注解实现跨域无效常见原因:
- springMvc 版本低于4.2X。
@RequestMapping
注解中没有给出请求方式。
@CrossOrigin
的参数:
- value属性可以设置多个URL。
- origins属性也可以设置多个URL。
- maxAge属性指定了准备响应前的缓存持续的最大时间。就是探测请求的有效期。
- allowCredentials属性表示用户是否可以发送、处理 cookie。默认为false
- allowedHeaders 属性表示允许的请求头部有哪些。
- methods 属性表示允许请求的方法,默认get,post,head。
springMvc 处理一个请求的顺序是: 过滤器 -> 拦截器 -> 最后才是控制器controller
拦截器重写示例:
isLoginInterceptor.java
public class isLoginInterceptor implements HandlerInterceptor {
//在请求处理的方法之前执行
//如果返回true执行下一个拦截器
//如果返回false就不执行下一个拦截器
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
System.out.println("---------------处理前---------------");
// System.out.println(request.getHeader(HttpHeaders.ORIGIN) + "origin");
// System.out.println(request.getMethod());
if (request.getHeader(HttpHeaders.ORIGIN) != null) {
//支持跨域这里手动为相应包添加允许跨域的属性,只对浏览器的options预先请求生效
response.addHeader("Access-Control-Allow-Origin", request.getHeader(HttpHeaders.ORIGIN));// origin 为请求接口的站点主域名
response.addHeader("Access-Control-Allow-Credentials", "true");
response.addHeader("Access-Control-Allow-Methods", "POST, GET, OPTIONS, DELETE, PUT, HEAD");
response.addHeader("Access-Control-Allow-Headers", "Content-Type");
response.addHeader("Access-Control-Max-Age", "3600");
return true;
}
User user = (User) request.getSession(false).getAttribute("USER_SESSION");
System.out.println(user + "interceptor");
if (user == null) {
System.out.println("user == null ");
return false;
} else
return true;
}
//在请求处理方法执行之后执行
public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
System.out.println("---------------处理后---------------");
}
//在dispatcherServlet处理后执行,做清理工作
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
System.out.println("---------------清理---------------");
}
}
springmvc-servlet.xml,这个是springMvc版xml配置拦截器。
<mvc:interceptors>
<mvc:interceptor>
<mvc:mapping path="/**">
<mvc:exclude-mapping path="/user"/>
<!--填写之前配置好的拦截器-->
<bean class="interceptor.isLoginInterceptor"></bean>
</mvc:interceptor>
</mvc:interceptors>
InterceptorConfig.java,这个是springBoot版注解配置拦截器。
@Configuration
public class InterceptorConfig implements WebMvcConfigurer {
/**
* 注册自定义拦截器
*/
@Override
public void addInterceptors(InterceptorRegistry registry) {
registry.addInterceptor(new Interceptor01()).addPathPatterns("/**").excludePathPatterns("/user/login");
}
}
request.setHeader
和request.addHeader
的不同点
request.setHeader
是在已有的属性上修改,如果没有会创建。
request.addHeader
是在header请求头里添加属性,不考虑属性会重复。
过滤器重写示例:
// 由于我没有测试成功,在这里就不贴出代码了。