在拦截器中使用threadlocal进行登录校验

一,编写UserThreadLocal类

public class UserThreadLocal {

    private static final ThreadLocal<User> LOCAL = new ThreadLocal<>();

    private UserThreadLocal(){

    }

    /**
     * 将对象放入到ThreadLocal
     *
     * @param user
     */
    public static void set(User user){
        LOCAL.set(user);
    }

    /**
     * 返回当前线程中的User对象
     *
     * @return
     */
    public static User get(){
        return LOCAL.get();
    }

    /**
     * 删除当前线程中的User对象
     */
    public static void remove(){
        LOCAL.remove();
    }

}

 二,编写tokenInterCepter

package com.tanhua.server.interceptor;

@Component
public class UserTokenIntercepter implements HandlerInterceptor {

    @Autowired
    private UserService userService;

    @Override
    public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
        //校验handler是否是handlerMethod
        if (!(handler instanceof MethodHandle)) {
            return true;
        }
        //判断是否包含noAuthorization注解,如果包含,直接放行
        if (((HandlerMethod)handler).hasMethodAnnotation(NoAuthorization.class)) {
            return true;
        }
        //如果没有,从请求头中获取token信息放入threadLocal
        String token = request.getHeader("Authorization");
        if (StrUtil.isNotEmpty(token)) {
            User user = this.userService.queryUserByToken(token);
            if (user != null) {
                UserThreadLocal.set(user);
                return true;
            }
        }
        response.setStatus(401);
        return false;
    }

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

    }
}

 三,编写注解NoAuthorization

@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
@Documented //标记注解
public @interface NoAuthorization {

}

四,注册拦截器

@Configuration
public class WebConfig implements WebMvcConfigurer {

    @Autowired
    private RedisCacheInterceptor redisCacheInterceptor;

    @Autowired
    private UserTokenIntercepter userTokenIntercepter;

    @Override
    public void addInterceptors(InterceptorRegistry registry) {
        registry.addInterceptor(this.userTokenIntercepter).addPathPatterns("/**");
        registry.addInterceptor(this.redisCacheInterceptor).addPathPatterns("/**");
    }
}

 五,使用threadLocal


public PageResult queryPublishList(Integer page, Integer pageSize) {
        PageResult pageResult = new PageResult();
        pageResult.setPage(page);
        pageResult.setPagesize(pageSize);

        //获取User对象,无需对User对象校验,其一定不为null
        User user = UserThreadLocal.get();

        PageInfo<Publish> pageInfo = this.quanZiApi.queryPublishList(user.getId(), page, pageSize);
        
       //。。。。代码略。。。。。

        return pageResult;
    }

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
拦截使用 `ThreadLocal` 存储数据时,需要注意以下几点: 1. `ThreadLocal` 的作用域是线程级别的,即同一个线程的所有代码都可以访问该 `ThreadLocal` 对象存储的数据,但不同线程之间是互不干扰的。 2. 在拦截,如果使用 `ThreadLocal` 存储数据,需要在请求处理结束后将其清空,否则该线程在下一次请求时可能会访问到上一次请求存储的数据。 3. 在控制访问 `ThreadLocal` 存储的数据时,需要确保在同一个线程,否则会出现获取到 `null` 的情况。 以下是一个示例代码: ```java public class MyInterceptor implements HandlerInterceptor { private static final ThreadLocal<String> threadLocal = new ThreadLocal<>(); @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { String value = // 获取需要存储的数据 threadLocal.set(value); return true; } @Override public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception { threadLocal.remove(); // 清空 ThreadLocal 的数据 } } @RestController public class MyController { @GetMapping("/test") public String test() { String value = MyInterceptor.threadLocal.get(); return "value: " + value; } } ``` 在上面的代码,`MyInterceptor` 使用 `ThreadLocal` 存储数据,`MyController` 通过访问 `MyInterceptor.threadLocal` 获取存储的数据。请注意,`MyInterceptor.threadLocal` 是一个静态变量,因此在不同的线程访问到的是不同的 `ThreadLocal` 对象。如果在 `MyController` 访问时获取到的是 `null`,那么很可能是因为在不同的线程访问了 `ThreadLocal` 对象。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值