java拦截器定义_轻松实现java拦截器+自定义注解

本文介绍如何在 SpringBoot 项目中实现自定义拦截器,并通过简洁的代码示例展示如何进行 token 验证,同时介绍了如何利用注解灵活配置免验证路径。

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

目录和概述

概述

内容:拦截器的使用很简单,定义一个自己的拦截器,向配置中添加一下就可以使用。为了方便,之后又引入了注解。本文就将用简洁的代码构建一个springboot的拦截器。

假设需求:访问项目的controller是都要进行"token验证",除了某些像登录之类的方法。

项目结构:

TokenInterceptor.java 自定义拦截器

InterceptorConfig.java 添加拦截器进入项目

NoNeedToken.java 自定义注解

TestController.java 测试接口

ad23b6a75294a82204d529e83468bf47.png

目录

1、自定义拦截器

在 TokenInterceptor.java 中输入以下代码,以下的代码将生成一个在请求到达controller前进行拦截的拦截器

import com.alibaba.fastjson.JSONObject;

import org.springframework.lang.Nullable;

import org.springframework.stereotype.Component;

import org.springframework.web.servlet.HandlerInterceptor;

import org.springframework.web.servlet.ModelAndView;

import javax.servlet.http.HttpServletRequest;

import javax.servlet.http.HttpServletResponse;

import java.io.PrintWriter;

import java.util.Arrays;

import java.util.List;

@Component

public class TokenInterceptor implements HandlerInterceptor {

// 假设现在的token有如下数据

List tokenList = Arrays.asList("111", "222", "333");

// 这个方法是在访问接口之前执行的,我们只需要在这里写验证登陆状态的业务逻辑,就可以在用户调用指定接口之前验证登陆状态了

@Override

public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {

// 设置返回为json格式,使用UTF-8

response.setCharacterEncoding("UTF-8");

response.setContentType("application/json; charset=utf-8");

String token = request.getHeader("token");

PrintWriter out;

// 之后写你的判断逻辑:return true是通过拦截器,可以继续访问controller,return false是不通过

if (token == null || !tokenList.contains(token)) {

// 如果失败了返回{state:"false", msg:"token is null or wrong"}

JSONObject res = new JSONObject();

res.put("state","false");

res.put("msg","token is null or wrong");

out = response.getWriter();

out.append(res.toString());

return false;

}

// 否则返回true 进入controller

return true;

}

public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, @Nullable ModelAndView modelAndView) throws Exception {

}

public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, @Nullable Exception ex) throws Exception {

}

}

2、载入拦截器

将以下代码写在InterceptorConfig.java中 ,使用addInterceptors方法将我们定义的拦截器添加进入项目中

import org.springframework.context.annotation.Configuration;

import org.springframework.web.servlet.config.annotation.InterceptorRegistry;

import org.springframework.web.servlet.config.annotation.WebMvcConfigurer;

import javax.annotation.Resource;

@Configuration

public class InterceptorConfig implements WebMvcConfigurer {

@Resource

TokenInterceptor tokenInterceptor;

@Override

public void addInterceptors(InterceptorRegistry registry) {

// 设置所有的路径都要进行拦截,除了/test/login

registry.addInterceptor(tokenInterceptor).addPathPatterns("/**")

.excludePathPatterns("/test/login");

}

}

3、测试成功

将如下代码到TestController中,用于测试拦截器效果

import org.springframework.web.bind.annotation.RequestMapping;

import org.springframework.web.bind.annotation.RestController;

import java.util.HashMap;

import java.util.Map;

@RestController

// canload用于检测拦截器是否正常工作

@RequestMapping("/test")

public class TestController {

@RequestMapping(value = "/canload")

public Map canLoad(){

Map res = new HashMap<>();

res.put("state", "ok");

res.put("msg", "you can load");

return res;

}

// login模拟登录时token返回

@RequestMapping(value = "/login")

public String login(){

return "111";

}

}

可以看到如下结果

①直接登录成功

b225810f7e784591af8902740875ace9.png

②直接访问canload失败

d659a369f102ca64ecabddde76c4e138.png

③使用postman,带上token进行访问成功

a70e0273935ad17c2a97fa9f99bc6714.png

4、使用注解(Annotation)进行选择

你也可以使用注解来代替第三步配置中的excludePathPatterns(), 来实现对于指定方法和controller的免token访问,新建一个NoNeedToken.java注解,在拦截器里判断注解存在时,直接return true放行

NoNeedToken.java

import java.lang.annotation.ElementType;

import java.lang.annotation.Retention;

import java.lang.annotation.RetentionPolicy;

import java.lang.annotation.Target;

@Target({ElementType.METHOD, ElementType.TYPE}) //注解的范围是类、接口、枚举的方法上

@Retention(RetentionPolicy.RUNTIME)//被虚拟机保存,可用反射机制读取

public @interface NoNeedToken {

}

响应的更改自定义拦截器的 preHandle方法,和拦截器配置

① 自定义拦截器

public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {

response.setCharacterEncoding("UTF-8");

response.setContentType("application/json; charset=utf-8");

String token = request.getHeader("token");

PrintWriter out;

// 对于注解的判断---------------------------------------------------

HandlerMethod handlerMethod = (HandlerMethod) handler;

if(handlerMethod.getMethodAnnotation(NoNeedToken.class)!=null || handlerMethod.getBeanType().isAnnotationPresent(NoNeedToken.class)){

// 如果自己拥有NoNeedToken标注或者所属的class拥有NoNeedToken 就直接放行

return true;

}

//------------------------------------------------------------------

if (token == null || !tokenList.contains(token)) {

JSONObject res = new JSONObject();

res.put("state","false");

res.put("msg","token is null or wrong");

out = response.getWriter();

out.append(res.toString());

return false;

}

return true;

}

② 去除拦截器配置中的去除项

public void addInterceptors(InterceptorRegistry registry) {

registry.addInterceptor(tokenInterceptor).addPathPatterns("/**");

// .excludePathPatterns("/test/login");

}

③ 在测试的login方法上添加注释

@RequestMapping(value = "/login")

@NoNeedToken

public String login(){

return "111";

}

修改完成之后效果和上面相同

5、常见问题

1.这个自定义注解的哪些@Target,@Retention,@Documented是什么意思

2.为什么我的注解加了,还是被拦截了

一般出现这种情况是类与方法的获取注解方式不同,在本文中作者使用了下面的一句话来表述

handlerMethod.getMethodAnnotation(NoNeedToken.class)!=null || handlerMethod.getBeanType().isAnnotationPresent(NoNeedToken.class)

其中handlerMethod.getMethodAnnotation(NoNeedToken.class)!=null是对于方法的

而handlerMethod.getBeanType().isAnnotationPresent(NoNeedToken.class)是对于类的

结语:希望对大家有帮助

Java拦截器是一种应用程序设计模式,它可以截获并处理请求、响应和异常。Spring框架提供了一个拦截器接口HandlerInterceptor,让我们可以在请求到达Controller之前或者Controller返回结果之后进行一些自定义的操作,如权限控制、日志记录、性能监控等。 Spring拦截器实现步骤如下: 1.定义一个类实现HandlerInterceptor接口,并实现其中的三个方法:preHandle、postHandle和afterCompletion。这些方法分别对应请求处理前、请求处理后和请求完成后的操作。 2.在Spring的配置文件中配置拦截器,并指定拦截的路径。可以使用<mvc:interceptors>标签来配置拦截器,也可以使用@Bean注解来配置拦截器。 3.在Controller类中使用@RequestMapping注解来指定请求路径。 4.启动应用程序,访问配置的路径,观察拦截器的执行效果。 HandlerInterceptor接口提供了三个方法,分别是: 1.preHandle方法:在请求处理之前进行调用,返回值为boolean类型。如果返回true,则继续向下执行;如果返回false,则不会执行Controller中的方法。 2.postHandle方法:在请求处理之后进行调用,但是在视图被渲染之前。即Controller方法调用之后,但是DispatcherServlet向客户端响应视图之前执行。可以利用ModelAndView参数来传递数据。 3.afterCompletion方法:在整个请求完成之后,也就是视图渲染之后执行。一般用于资源清理操作。 在Spring框架中,拦截器的执行顺序与它们在配置文件中的顺序有关。如果有多个拦截器,那么它们的执行顺序是按照配置文件中定义的顺序依次执行的。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值