1. 自定义注解

​ 当用户在查询的时候,要求用户必须是登录状态。就应该让用户登录,所以在此我们自定义一个注解来表示访问此功能时必须要登录。

注解作用:哪些需要登录才能访问必须要添加,那些需要获取到用户Id控制层方法也必须加这个注解.


import java.lang.annotation.*;

/**
 * 自定义注解:该注解被使用到web层controller方法上,用与验证用户认证状态
 * 元注解:
 * @Target:注解使用位置,可选:类/接口/注解、方法、属性、方法参数、构造等
 * @Retention:注解声明周期(注解会保留到什么阶段)可选:SOURCE、CLASS、RUNNTIME 选择CLASS该注解在运行时没了
 * @Inherited:该注解是否可以被继承
 * @Documented:通过JDK提供javadoc命令产生文档,是否会生成该注解文档
 */
@Target({ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Inherited
@Documented
public @interface Login {

    /**
     * 是否必须登录,默认为:必须登录
     * @return
     */
    boolean required() default true;


}

1.1 切面类

RequestContextHolder类持有上下文的Request容器。

RequestContextHolder是Spring框架提供的一个用于管理请求上下文的工具类。它允许在应用程序的任何地方访问当前的HTTP请求和响应信息,而不需要将这些对象显式地传递给每个需要它们的方法或对象

request和response等是什么时候设置进去的?

springMVC 核心类DispatcherServlet 继承关系

自定义切面类

import com.atguigu.tingshu.common.constant.RedisConstant;
import com.atguigu.tingshu.common.execption.GuiguException;
import com.atguigu.tingshu.common.result.ResultCodeEnum;
import com.atguigu.tingshu.common.util.AuthContextHolder;
import com.atguigu.tingshu.vo.user.UserInfoVo;
import jakarta.servlet.http.HttpServletRequest;
import jakarta.servlet.http.HttpServletResponse;
import lombok.SneakyThrows;
import lombok.extern.slf4j.Slf4j;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.stereotype.Component;
import org.springframework.web.context.request.RequestAttributes;
import org.springframework.web.context.request.RequestContextHolder;
import org.springframework.web.context.request.ServletRequestAttributes;


@Slf4j
@Aspect //声明切面类
@Component //将切面类对象注册到IOC容器
public class LoginAspect {

    @Autowired
    private RedisTemplate redisTemplate;

    /**
     * 验证登录状态、获取用户ID通知逻辑
     *
     * @param joinPoint  切点方法对象
     * @param Login 自定义登录注解
     * @return
     * execution(* com.包名.包名.*.api.*.*(..)) && @annotation(Login)
     *  这是一个AOP(面向切面编程)的注解,用于定义一个环绕通知。它表示在执行com.包名.包名 包下的所有类      *    中,所有带有@Login注解的方法时,都会触发这个环绕通知。
     */
    @SneakyThrows //动态生成try catch(Throwable)
    @Around("execution(* com.atguigu.tingshu.*.api.*.*(..)) && @annotation(guiGuLogin)")
    public Object loginAspect(ProceedingJoinPoint joinPoint, GuiGuLogin guiGuLogin) {
        Object resultObject = new Object();
        log.info("前置通知...");

        //1.获取小程序端(客户端)提交令牌Token
        //1.1 通过请求上下文对象RequestContextHolder获取请求属性对象 - 普通类中获取到请求对象
        RequestAttributes requestAttributes = RequestContextHolder.getRequestAttributes();
        //1.2 将RequestAttributes(接口)转为ServletRequestAttributes(实现类)
        ServletRequestAttributes servletRequestAttributes = (ServletRequestAttributes) requestAttributes;
        //1.3 获取请求对象
        HttpServletRequest request = servletRequestAttributes.getRequest();
        //HttpServletResponse response = servletRequestAttributes.getResponse();

        //1.4 基于请求对象获取请求头:token
        String token = request.getHeader("token");

        //2.查询存放在Redis中用户信息(何时存入:登录成功后)
        //2.1 构建用户登录信息key
        String loginKey = RedisConstant.USER_LOGIN_KEY_PREFIX + token;
        //2.2 查询存入Redis中用户信息
        UserInfoVo UserInfoVo = (UserInfoVo) redisTemplate.opsForValue().get(loginKey);

        //3.判断注解是否要求必须登录,如果要求登录且用户信息为空,抛出异常业务状态:208引导用户登录
        if (Login.required() && UserInfoVo == null) {
            //抛异常
        }

        //4.如果用户信息存在,将用户ID存入ThreadLocal
        if (UserInfoVo != null) {
            AuthContextHolder.setUserId(UserInfoVo.getId());
        }
        //二、执行目标方法
        resultObject = joinPoint.proceed();

        //5.将ThreaLocal中数据手动清理避免OOM出现
        AuthContextHolder.removeUserId();
        log.info("后置通知...");
        return resultObject;
    }
}

在查看专辑列表的时候,添加注解。在点击查看专辑列表的时候,就会提示我们需要进行登录!

1.2 使用认证校验注解

在所有业务微服务中,com.包名.包名.*.api.XXXXController类中下所有需要登录才可以访问方法上加注解@Login 以下接口测试需要完成登录代码后才进行测试