Springboot+Aop用注解实现阿里云短信验证码校验,校验通过自动删除验证码缓存

1.新建操作类型枚举(这里的IEnum是我自定义的http请求拦截接口,不需要的话可以不用实现)

java 复制代码
@Getter
@AllArgsConstructor
public enum OperationType implements IEnum<Integer> {
    /**
     * 注册
     */
    SIGN_UP(0),
    /**
     * 密码登录
     */
    LOGIN_BY_PWD(1),
    /**
     * 验证码登录
     */
    LOGIN_BY_SMS(2),
    /**
     * 忘记密码
     */
    FORGET_PWD(3),
    /**
     * 修改密码
     */
    MODIFY_PWD(4);

    @JsonValue
    private final int code;

    @Override
    public Integer getCode(){
        return code;
    }
}

2.新建校验注解

java 复制代码
@Target({ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface SmsValidate {
    /**
     * 操作类型
     * @return
     */
    OperationType operationType() default OperationType.LOGIN_BY_PWD;
}

3.创建验证码校验接口

java 复制代码
public interface ISmsValidate {
    /**
     * 短信验证码手机号
     * @return
     */
    String mobile();

    /**
     * 短信验证码内容
     * @return
     */
    String smsCode();
}

4.controller方法请求参数,实现ISmsValidate接口

java 复制代码
@Data
public class LoginBySmsDto implements ISmsValidate {
    @NotBlank(message = "用户名/手机号不能为空")
    @IsMobile
    private String username;
    @NotBlank(message = "短信验证码不能为空")
    private String code;

    @Override
    public String mobile() {
        return this.getUsername();
    }

    @Override
    public String smsCode() {
        return this.getCode();
    }
}

5.添加aop切面类

java 复制代码
@Aspect
@Component
public class SmsValidateAop {
    @Autowired
    private RedisTemplate<String,Object> redisTemplate;

    @Pointcut(value = "@annotation(com.tfyt.common.annotation.SmsValidate)")
    public void pointCut(){}

    @Before(value = "pointCut()")
    public void before(JoinPoint joinPoint){
        SmsObj smsObj = getSmsObj(joinPoint);
        Object cacheCode = redisTemplate.opsForValue().get(RedisKeyConstant.CACHE_SMS_CODE + smsObj.getOperationType() + ":" + smsObj.getMobile());
        BusinessAssert.notTrue(Objects.equals(cacheCode,smsObj.getCode()),"手机验证码不正确");
    }

    @AfterReturning(value = "pointCut()")
    public void after(JoinPoint joinPoint){
        SmsObj smsObj = getSmsObj(joinPoint);
        redisTemplate.delete(RedisKeyConstant.CACHE_SMS_CODE + smsObj.getOperationType() + ":" + smsObj.getMobile());
    }

    @Data
    @AllArgsConstructor
    @NoArgsConstructor
    private static class SmsObj{
        private String mobile;
        private String code;
        private Integer operationType;
    }

    private SmsObj getSmsObj(JoinPoint joinPoint){
        MethodSignature signature = (MethodSignature) joinPoint.getSignature();
        SmsValidate annotation = signature.getMethod().getAnnotation(SmsValidate.class);
        BusinessAssert.isNull(annotation,"系统异常:未查询到注解");
        BusinessAssert.isNull(annotation.operationType(),"系统异常:未配置操作类型");
        Object[] args = joinPoint.getArgs();
        ISmsValidate arg = null;
        if(args[0] instanceof ISmsValidate){
            arg = (ISmsValidate) args[0];
        }
        BusinessAssert.isNull(arg,"请输入用户名和手机验证码");
        return new SmsObj(arg.mobile(), arg.smsCode(), annotation.operationType().getCode());
    }
}

6.往controller的方法上注解

复制代码
@SmsValidate(operationType = OperationType.LOGIN_BY_SMS)

重启springboot项目,调用接口即可生效,校验通过后会自动删除redis缓存

补一个自定义断言类

java 复制代码
public class BusinessAssert {

    public static void notTrue(boolean condition, String msg){
        isTrue(!condition, msg);
    }

    public static void isTrue(boolean condition, String msg){
        if(condition){
            throw new BusinessException(HttpStatus.BAD_REQUEST.value(),msg);
        }
    }

    public static void nonNull(Object object,String msg){
        if(null!=object){
            throw new BusinessException(HttpStatus.BAD_REQUEST.value(),msg);
        }
    }

    public static void isNull(Object object,String msg){
        if(null==object){
            throw new BusinessException(HttpStatus.BAD_REQUEST.value(),msg);
        }
    }

    public static void isCollectionEmpty(Collection<?> collection,String msg){
        if(collection == null || collection.isEmpty()){
            throw new BusinessException(HttpStatus.BAD_REQUEST.value(),msg);
        }
    }

    public static void isCollectionNotEmpty(Collection<?> collection,String msg){
        if(collection != null && !collection.isEmpty()){
            throw new BusinessException(HttpStatus.BAD_REQUEST.value(),msg);
        }
    }
相关推荐
葫芦和十三11 小时前
图解 MongoDB 05|文档模型设计:内嵌 vs 引用,反范式不是免费午餐
后端·mongodb·agent
不能放弃治疗14 小时前
单 Agent 实现模式
后端
IT_陈寒16 小时前
Redis内存爆了,原来我漏掉了这个致命配置
前端·人工智能·后端
小bo波17 小时前
从"任意文件复制"深挖Java I/O:字符流与字节流的本质抉择
java·nio·io流·后端开发·文件复制
fliter17 小时前
最后一块拼图:用 bitvec 构造 IPv4 包,真正做出自己的 Ping
后端
用户35218024547518 小时前
🎆从 Prompt 到 Skill:让 Spring AI Agent 学会"装新技能"
人工智能·spring boot·ai编程
fliter18 小时前
用 Rust 解析并生成 ICMP 包:checksum、nom 与 cookie-factory
后端
蝎子莱莱爱打怪18 小时前
XZLL-IM干货系列 03|消息 ID 设计:一个 UUID 搞不定的事,我用两个 ID 解决了
后端·面试·开源
fliter18 小时前
从 panic 到 Result:用 Rust 重新整理一个 ping 项目的错误处理
后端
森蓝情丶19 小时前
我给 AI 搭了个法庭:一个前端仔的 LangGraph 实战全记录
前端·后端