SpringBoot的@controller类,实现ajax跨域的方法.

本文详细介绍了四种解决跨域问题的方法:直接在控制器方法中设置响应头、使用拦截器、在控制器方法上使用@CrossOrigin注解以及利用Zuul的CorsFilter过滤器。每种方法都提供了具体的代码实现,便于开发者理解和应用。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

第一种

直接在@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过滤器

 

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值