spring防止重复点击,两种注解实现(AOP)

第一种:@EasyLock

简介

为了简化可复用注解,自己实现的注解,代码简单随拿随用

使用方式

1.创建一个注解

java 复制代码
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface EasyLock {
    long waitTime() default 1;
    long leaseTime() default 3;
}
  1. 创建一个AOP切面类(异常可以自定义,这里我用的Cicada)
java 复制代码
import lombok.extern.slf4j.Slf4j;
import org.apache.commons.lang.StringUtils;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.aspectj.lang.annotation.Pointcut;
import org.aspectj.lang.reflect.MethodSignature;
import org.redisson.api.RLock;
import org.redisson.api.RedissonClient;
import org.springframework.stereotype.Component;
import vip.lspace.agent.common.annotation.EasyLock;
import vip.lspace.agent.common.exception.LockException;

import javax.annotation.Resource;
import java.util.concurrent.TimeUnit;

@Component
@Aspect
@Slf4j
public class LockAop {
    @Resource
    private LockException lockException;

    @Resource
    RedissonClient redissonClient;

    private static final String redisLockKeyParamName = "redisLockKey";

    @Pointcut("@annotation(vip.lspace.agent.common.annotation.EasyLock)")
    public void lockPointcut() {
    }

    @Around("lockPointcut()")
    public Object doAround(ProceedingJoinPoint proceedingJoinPoint) {
        log.info("EasyLock locking");
        MethodSignature methodSignature = (MethodSignature) proceedingJoinPoint.getSignature();
        EasyLock easyLock = methodSignature.getMethod().getAnnotation(EasyLock.class);

        Object[] args = proceedingJoinPoint.getArgs();
        String[] parameterNames = methodSignature.getParameterNames();
        String redisLockKey = "";
        for (int i = 0; i < parameterNames.length; i++) {
            if (parameterNames[i].equals(redisLockKeyParamName)) {
                redisLockKey = (String) args[i];
            }
        }
        if (StringUtils.isBlank(redisLockKey)) {
            throw lockException.lockKeyNotExist();
        }

        RLock lock = redissonClient.getLock(redisLockKey);
        try {
            if (!lock.tryLock(easyLock.waitTime(), easyLock.leaseTime(), TimeUnit.SECONDS)) {
                throw lockException.getLockTimeOut(redisLockKey);
            }
            return proceedingJoinPoint.proceed();
        } catch (Throwable e) {
            throw new RuntimeException(e);
        } finally {
            if (lock.isLocked() && lock.isHeldByCurrentThread()) {
                lock.unlock();
                log.info("当前线程{},释放锁:{}", Thread.currentThread().getId(), redisLockKey);
            }
        }
    }
java 复制代码
@CicadaBean(namespace = "lock")
public interface LockException {
    @ExceptionInfo(errCode = 13001, errMessage = "没找到分布式锁key")
    CicadaException lockKeyNotExist();

    @ExceptionInfo(errCode = 13002, errMessage = "超时未获取到锁: %s")
    CicadaException getLockTimeOut(String key);
}

3.使用方式

注意点:

必须包含名为redisLockKey的参数,作为redis的key

java 复制代码
@Override
    @EasyLock(waitTime = 1,
            leaseTime = 3
    )
    @Transactional(rollbackFor = Exception.class)
    public void concurrentTest(String redisLockKey) {
        PaymentBillBacktrack paymentBillBacktrack = paymentBillBacktrackService.getById(1L);
        String refundRequestNo = paymentBillBacktrack.getMerchantRefundRequestNo();
        paymentBillBacktrack.setMerchantRefundRequestNo(String.valueOf(Integer.parseInt(refundRequestNo) + 1));
        paymentBillBacktrackService.updateById(paymentBillBacktrack);
    }

第二种:@NiceLock

简介

大佬提供的公共组件,引包后可直接使用,使用简单,细节代码可看nicelock: nicelock:一个注解,即可使用Java的分布式锁。(基于Redisson)

使用方式

1.引包

一定得引入1.1.6的,甚至最新版本,不然有问题!!!

XML 复制代码
<dependency>
     <groupId>com.suchtool</groupId>
     <artifactId>nicelock-spring-boot-starter</artifactId>
     <version>1.1.6</version>
</dependency>

2.使用

java 复制代码
    @Override
    @Transactional
    @NiceLock(
            keys = {"#userId"},
            acquireTimeout = 3000L,
            exception = NiceLockLockFailException.class,
            message = "服务已完成评价,不能重复提交"
    )
    public void test(String userId) {
        System.out.println("修改订单: 用户ID=" + userId);
    }

注解中传入了exception参数后,报错会输出message的内容,更加直观

测试方式

Apifox中的自动化测试中可以配多个线程同时执行接口,测试重复点击是否生效

相关推荐
xuxie991 天前
N11 ARM-irq
java·开发语言
cjy0001111 天前
springboot的 nacos 配置获取不到导致启动失败及日志不输出问题
java·spring boot·后端
wefly20171 天前
从使用到原理,深度解析m3u8live.cn—— 基于 HLS.js 的 M3U8 在线播放器实现
java·开发语言·前端·javascript·ecmascript·php·m3u8
zhenxin01221 天前
Spring Boot实现定时任务
java
小江的记录本1 天前
【事务】Spring Framework核心——事务管理:ACID特性、隔离级别、传播行为、@Transactional底层原理、失效场景
java·数据库·分布式·后端·sql·spring·面试
sheji34161 天前
【开题答辩全过程】以 基于springboot的校园失物招领系统为例,包含答辩的问题和答案
java·spring boot·后端
寂静or沉默1 天前
2026最新Java岗位从P5-P7的成长面试进阶资源分享!
java·开发语言·面试
卓怡学长1 天前
m289在线交友系统
java·spring·tomcat·maven·intellij-idea·hibernate
程序员cxuan1 天前
人麻了,谁把我 ssh 干没了
人工智能·后端·程序员
zhglhy1 天前
Java分库分表技术对比分析
java·分库分表