第一种
直接在@Controller类下的方法里使用
response.addHeader("Access-Control-Allow-Origin", "*");
response.setHeader("Content-Type", "application/json;charset=UTF-8");
如下:
@GetMapping("/font")
@ResponseBody
public long font(HttpServletResponse response){
response.addHeader("Access-Control-Allow-Origin", "*");
response.setHeader("Content-Type", "application/json;charset=UTF-8");
return 200;
}
第二种
使用拦截器,Interceptor类代码如下:
@Slf4j
public class Interceptor implements HandlerInterceptor {
/**
* preHandle方法是进行处理器拦截用的,顾名思义,该方法将在Controller处理之前进行调用,SpringMVC中的Interceptor拦截器是链式的,可以同时存在
* 多个Interceptor,然后SpringMVC会根据声明的前后顺序一个接一个的执行,而且所有的Interceptor中的preHandle方法都会在
* Controller方法调用之前调用。SpringMVC的这种Interceptor链式结构也是可以进行中断的,这种中断方式是令preHandle的返
* 回值为false,当preHandle的返回值为false的时候整个请求就结束了。
*/
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
response.setHeader("Access-Control-Allow-Origin", request.getHeader("Origin"));
response.setHeader("Access-Control-Allow-Methods", "*");
response.setHeader("Access-Control-Allow-Credentials", "true");
response.setHeader("Access-Control-Allow-Headers", "x-requested-with,content-type");
System.err.println("------------------>:已完成跨域处理");
long startTime = System.currentTimeMillis();
request.setAttribute("startTime", startTime);
return true;
}
/**
* 这个方法只会在当前这个Interceptor的preHandle方法返回值为true的时候才会执行。postHandle是进行处理器拦截用的,它的执行时间是在处理器进行处理之
* 后,也就是在Controller的方法调用之后执行,但是它会在DispatcherServlet进行视图的渲染之前执行,也就是说在这个方法中你可以对ModelAndView进行操
* 作。这个方法的链式结构跟正常访问的方向是相反的,也就是说先声明的Interceptor拦截器该方法反而会后调用,这跟Struts2里面的拦截器的执行过程有点像,
* 只是Struts2里面的intercept方法中要手动的调用ActionInvocation的invoke方法,Struts2中调用ActionInvocation的invoke方法就是调用下一个Interceptor
* 或者是调用action,然后要在Interceptor之前调用的内容都写在调用invoke之前,要在Interceptor之后调用的内容都写在调用invoke方法之后。
*/
public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, @Nullable ModelAndView modelAndView) throws Exception {
long startTime = (Long) request.getAttribute("startTime");
long endTime = System.currentTimeMillis();
long executeTime = endTime - startTime;
//修改存在的modelAndView
if ("public java.lang.String com.example.demo.controller.InterceptorController.jump()".equals(handler.toString())) {
modelAndView.addObject("executeTime", executeTime);
}
log.info("[" + handler + "] executeTime : " + executeTime + "ms");
}
/**
* 该方法也是需要当前对应的Interceptor的preHandle方法的返回值为true时才会执行。该方法将在整个请求完成之后,也就是DispatcherServlet渲染了视图执行,
* 这个方法的主要作用是用于清理资源的,当然这个方法也只能在当前这个Interceptor的preHandle方法的返回值为true时才会执行。
*/
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, @Nullable Exception ex) throws Exception {
System.out.println("执行afterCompletion方法,可在此处释放资源");
}
}
AppConfigurer类如下:
@Configuration
public class AppConfigurer extends WebMvcConfigurationSupport {
@Bean
public HandlerInterceptor getMyInterceptor() {
return new Interceptor();
}
@Override
public void addInterceptors(InterceptorRegistry registry) {
// 注册ecm系统操作日志拦截器(凡加了自定义注解的方法都会添加用户操作日志),拦截所有请求
registry.addInterceptor(getMyInterceptor()).addPathPatterns("/**");
// 注册ecm系统请求拦截器(不拦截登录、注销、回调请求,其他都拦截)
registry.addInterceptor(new Interceptor()).addPathPatterns("/**");
// 连接器里没有@Autowire或者@Resource注入的属性就这么用
super.addInterceptors(registry);
}
}
这样项目启动后,访问@Controller类,拦截器就会触发。
控制台会打印
------------------>:已完成跨域处理
第三种:
在controller类的方法上使用@CrossOrigin
第四种:
使用zuul的CorsFilter过滤器