亚洲激情专区-91九色丨porny丨老师-久久久久久久女国产乱让韩-国产精品午夜小视频观看

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Java-Redis-Redisson分布式鎖的功能如何使用及實現

發布時間:2022-08-03 14:56:19 來源:億速云 閱讀:145 作者:iii 欄目:開發技術

今天小編給大家分享一下Java-Redis-Redisson分布式鎖的功能如何使用及實現的相關知識點,內容詳細,邏輯清晰,相信大部分人都還太了解這方面的知識,所以分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后有所收獲,下面我們一起來了解一下吧。

前置

Java-Redis-Redisson配置基礎上我們進行了改造,讓鎖的使用更加方便

基礎設施

RedissonLock

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
@Target(ElementType.METHOD)
@Retention(RetentionPolicy.RUNTIME)
public @interface RedissonLock {
    int lockTime() default 3; //加鎖的時間默認3秒,  如果任務在3秒內執行完畢那么自動釋放鎖,如果任務3秒內沒有執行完畢也會釋放鎖, 所以內容執行時間過長適當加大鎖的時間
    String key() default "" ;  //唯一標識,如果沒有那么默認為token->sessionId
    String doc() default "重復提交請求,請稍后再試";
    boolean repeatLock() default false; //可重復加鎖直到加鎖成功,默認為false不能重復加鎖
    int repeatLockCount() default -1; //可重復加鎖限制加鎖的次數, 默認-1直到成功,設置10那么加鎖10次都沒成功就直接返回
    int lockWaitTimeMs() default 100; //重復加鎖默認的阻塞時間100毫秒,可以自己定義
}

RepeatSubmitAspect

import com.application.Result;
import com.commonutils.NullUtils;
import com.redis.utils.DistributedRedisLock;
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.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.stereotype.Component;
import org.springframework.util.Assert;
import org.springframework.web.context.request.RequestContextHolder;
import org.springframework.web.context.request.ServletRequestAttributes;
import javax.servlet.http.HttpServletRequest;
@Aspect
@Component
public class RepeatSubmitAspect {
    @Value("${spring.redis.redisson.tokenName}")
    private  String tokenName;
    @Autowired
    private DistributedRedisLock redisLock;
    @Pointcut("@annotation(noRepeatSubmit)")
    public void pointCut(RedissonLock noRepeatSubmit) {
    }
    public static HttpServletRequest getRequest() {
        ServletRequestAttributes ra= (ServletRequestAttributes) RequestContextHolder.getRequestAttributes();
        return ra.getRequest();
    }
    @Around("pointCut(noRepeatSubmit)")
    public Object around(ProceedingJoinPoint pjp, RedissonLock noRepeatSubmit) throws Throwable {
        int lockSeconds = noRepeatSubmit.lockTime();
        String doc = noRepeatSubmit.doc();
        String keyName = noRepeatSubmit.key();
        boolean b = noRepeatSubmit.repeatLock();
        int repeatLockCount = noRepeatSubmit.repeatLockCount();
        int lockWaitTimeMs = noRepeatSubmit.lockWaitTimeMs();
        HttpServletRequest request = getRequest();
        Assert.notNull(request, "request can not null");
        //如果沒有唯一表示那么就使用token或者sessionID來唯一表示
        if(!NullUtils.notEmpty(keyName)){
            String token = request.getHeader(tokenName);
            if(NullUtils.notEmpty(token)){
                keyName=token;
            }else{
                //使用sessionID (注意保證分布式session共享)
                keyName = request.getSession().getId();
            }
            System.out.println("tokenName:"+keyName);
        }
        String path = request.getServletPath();
        String key = getKey(keyName, path);
        //加鎖
        boolean isSuccess = redisLock.acquire(key, lockSeconds,b,repeatLockCount,lockWaitTimeMs);
        if (isSuccess) {
            // 獲取鎖成功
            Object result;
            try {
                // 執行
                result = pjp.proceed();
            } finally {
                // 解鎖
                redisLock.release(key);
            }
            return result;
        } else {
            // 獲取鎖失敗,認為是重復提交的請求
            return Result.Error(doc);
        }
    }
    private String getKey(String token, String path) {
        return token + path;
    }
}

DistributedRedisLock

/**
 * 簡要描述
 *
 * @Author: huanmin
 * @Date: 2022/8/1 17:39
 * @Version: 1.0
 * @Description: 文件作用詳細描述....
 */

import com.multithreading.utils.SleepTools;
import lombok.SneakyThrows;
import org.redisson.api.RLock;
import org.redisson.api.RedissonClient;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Component;

import java.util.concurrent.TimeUnit;

@Component
public class DistributedRedisLock {
    private static final Logger logger = LoggerFactory.getLogger(DistributedRedisLock.class);
    //從配置類中獲取redisson對象
    @Autowired
    private RedissonClient redissonClient;
    private final String LOCK_TITLE = "redisLock_";
    private final ThreadLocal<Integer> count = new ThreadLocal<>();//計數

    //加鎖 , 線程加鎖的時候發現,鎖有人用了 ,那么就會進入自旋等待
    @SneakyThrows
    public boolean acquire(String lockKey, long seconds, boolean repeatLock, int repeatLockCount, int lockWaitTimeMs) {
        count.set(0); //初始化值
        //獲取鎖對象
        RLock mylock = redissonClient.getLock((LOCK_TITLE + lockKey));
        do {
            //嘗試加鎖
            boolean b = mylock.tryLock(0, seconds, TimeUnit.SECONDS);;
            if (b) {
                logger.info("獲取鎖成功");
                return true;
            }
            logger.info("嘗試獲取鎖" + Thread.currentThread().getName());
            //獲取加鎖的次數,如果是-1那么持續加鎖,如果滿足加鎖次數那么結束加鎖
            if (repeatLockCount != -1 && count.get().equals(repeatLockCount)) {
                logger.warn(Thread.currentThread().getName() + "嘗試加鎖失敗以嘗試了:" + count.get() + "次");
                return false;
            }
            SleepTools.ms(lockWaitTimeMs);
            //加鎖次數增加
            count.set(count.get() + 1);
        } while (repeatLock);
        logger.warn("重復提交");
        return false;
    }

    //手動鎖的釋放
    public void release(String lockKey) {
        //獲取所對象
        RLock mylock = redissonClient.getLock((LOCK_TITLE + lockKey));
        // 這里判斷下當前key是否上鎖,不然業務執行時間大于鎖自動釋放時間后,解鎖報異常
        if (mylock.isLocked()&&mylock.isHeldByCurrentThread()) { // 是否還是鎖定狀態并且鎖是當前線程的
                mylock.unlock(); // 釋放鎖
                logger.info("解鎖:" + lockKey);
        }

    }
}

功能使用和介紹

  • 支持配置鎖的時間

  • 支持配置鎖的key(同一個key的請求會被鎖住)

  • 支持重復加鎖

  • 支持重復加鎖的次數

  • 支持重復加鎖的間隔時間

通過以上功能的的組合能做到,冪等性,分布式悲觀鎖, 超時丟棄 ,不止在Controller里使用,而是在任何基于Spring容器管理的Bean都支持,當然如果特殊的場景我們可以直接使用DistributedRedisLock類,注意在任何時候都要釋放鎖

    /***
     *默認加鎖3秒
     * 加鎖的key為token->sessionId
     * 加鎖失敗不可重復加鎖
     * 默認提示:  重復提交請求,請稍后再試
     */
    @RedissonLock
    @GetMapping(value = "/updateAgeAsyncLock" )
    public Result updateAgeAsyncLock() {
        userService.updateAgeAsyncLock();
        return Result.Ok();
    }

    /***
     * 默認加鎖3秒
     * 加鎖失敗不可重復加鎖
     * 加鎖的key為token->sessionId
     * 提示:  加鎖失敗
     */
    @RedissonLock(doc = "加鎖失敗")
    @GetMapping(value = "/updateAgeAsyncLockDoc" )
    public Result updateAgeAsyncLockDoc() {
        userService.updateAgeAsyncLock();
        return Result.Ok();
    }
    /***
     * 默認加鎖15秒
     * 加鎖失敗不可重復加鎖
     * 加鎖的key為token->sessionId
     * 默認提示:  重復提交請求,請稍后再試
     */
    @RedissonLock(lockTime = 15)
    @GetMapping(value = "/updateAgeAsyncLock0" )
    public Result updateAgeAsyncLock0() {
         userService.updateAgeAsyncLock();
         return Result.Ok();
    }

    /***
     * 默認加鎖15秒
     * 加鎖失敗不可重復加鎖
     * 加鎖的key為updateAgeAsyncLock
     * 默認提示:  重復提交請求,請稍后再試
     */
    @RedissonLock(lockTime = 15,key = "updateAgeAsyncLock")
    @GetMapping(value = "/updateAgeAsyncLock1" )
    public Result updateAgeAsyncLock1() {
        userService.updateAgeAsyncLock();
        return Result.Ok();
    }
    /***
     * 默認加鎖15秒
     * 加鎖失敗可重復加鎖,直到加鎖成功
     * 加鎖的key為updateAgeAsyncLock
     * 默認提示:  重復提交請求,請稍后再試
     */
    @RedissonLock(lockTime = 15,key = "updateAgeAsyncLock",repeatLock = true)
    @GetMapping(value = "/updateAgeAsyncLock2" )
    public Result updateAgeAsyncLock2() {
        userService.updateAgeAsyncLock();
        return Result.Ok();
    }

    /***
     * 默認加鎖15秒
     * 加鎖失敗可重復加鎖10次,每次默認間隔100毫秒
     * 加鎖的key為updateAgeAsyncLock
     * 默認提示:  重復提交請求,請稍后再試
     */
    @RedissonLock(lockTime = 15,key = "updateAgeAsyncLock",repeatLock = true,repeatLockCount = 10)
    @GetMapping(value = "/updateAgeAsyncLock3" )
    public Result updateAgeAsyncLock3() {
        userService.updateAgeAsyncLock();
        return Result.Ok();
    }
    /***
     * 默認加鎖15秒
     * 加鎖失敗可重復加鎖10次,每次默認間隔500毫秒
     * 加鎖的key為updateAgeAsyncLock
     * 默認提示:  重復提交請求,請稍后再試
     */
    @RedissonLock(lockTime = 15,key = "updateAgeAsyncLock",repeatLock = true,repeatLockCount = 10,lockWaitTimeMs = 500)
    @GetMapping(value = "/updateAgeAsyncLock4" )
    public Result updateAgeAsyncLock4() {
        userService.updateAgeAsyncLock();
        return Result.Ok();
    }
    /***
     * 默認加鎖15秒
     * 加鎖失敗可重復加鎖,直到成功,每次嘗試間隔500毫秒
     * 加鎖的key為updateAgeAsyncLock
     * 默認提示:  重復提交請求,請稍后再試
     */
    @RedissonLock(lockTime = 15,key = "updateAgeAsyncLock",repeatLock = true,lockWaitTimeMs = 500)
    @GetMapping(value = "/updateAgeAsyncLock4" )
    public Result updateAgeAsyncLock4() {
        userService.updateAgeAsyncLock();
        return Result.Ok();
    }

驗證方式: 使用jmeter多線程10000請求 ,注意使用jmeter的時候不要使用默認鎖key的方式,因為jmeter每次請sessionID都不同的,想要驗證效果我們需要手動加key或者使用token

其他悲觀鎖的實現方式

public class DistributedRedisLock {
    //從配置類中獲取redisson對象
    @Autowired
    private RedissonClient redissonClient;
    private  final String LOCK_TITLE = "redisLock_";
    //加鎖 , 線程加鎖的時候發現,鎖有人用了 ,那么就會進入自旋等待
    public  boolean acquire(String lockName){
        //聲明key對象
        String key = LOCK_TITLE + lockName;

        //獲取鎖對象
        RLock mylock = redissonClient.getLock(key);
        //一直等待直到加鎖成功后,并且設置鎖過期時間,防止死鎖的產生
        mylock.lock(10, TimeUnit.SECONDS);
        System.err.println("======lock======"+Thread.currentThread().getName());
        //加鎖成功
        return  true;
    }
    //鎖的釋放
    public    void release(String lockName){
        //必須是和加鎖時的同一個key
        String key = LOCK_TITLE + lockName;
        //獲取所對象
        RLock mylock = redissonClient.getLock(key);
     // 這里判斷下當前key是否上鎖,不然業務執行時間大于鎖自動釋放時間后,解鎖報異常
        if(mylock.isLocked()){ // 是否還是鎖定狀態
            if(mylock.isHeldByCurrentThread()){ // 時候是當前執行線程的鎖
                mylock.unlock(); // 釋放鎖
                System.err.println("======unlock======"+Thread.currentThread().getName());
            }
        }
    }
}

以上就是“Java-Redis-Redisson分布式鎖的功能如何使用及實現”這篇文章的所有內容,感謝各位的閱讀!相信大家閱讀完這篇文章都有很大的收獲,小編每天都會為大家更新不同的知識,如果還想學習更多的知識,請關注億速云行業資訊頻道。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

阜康市| 丰县| 佛教| 广水市| 从江县| 仁寿县| 稻城县| 泗水县| 黑龙江省| 徐州市| 南皮县| 天峻县| 晋江市| 红桥区| 三台县| 新源县| 九台市| 夏津县| 禄丰县| 墨脱县| 金昌市| 芦溪县| 南乐县| 封开县| 沁阳市| 延吉市| 柳州市| 新津县| 江津市| 武平县| 徐汇区| 博乐市| 宜君县| 泰宁县| 阿克陶县| 阳东县| 榆树市| 温泉县| 湖南省| 都安| 定结县|