欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Java-Redis-Redisson分布式鎖的功能使用及實現(xiàn)

 更新時間:2022年08月03日 11:39:54   作者:胡安民  
這篇文章主要介紹了Java-Redis-Redisson-分布式鎖的功能使用及實現(xiàn),本文通過實例代碼給大家介紹的非常詳細,對大家的學(xué)習或工作具有一定的參考借鑒價值,需要的朋友可以參考下

前置

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

基礎(chǔ)設(shè)施

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秒,  如果任務(wù)在3秒內(nèi)執(zhí)行完畢那么自動釋放鎖,如果任務(wù)3秒內(nèi)沒有執(zhí)行完畢也會釋放鎖, 所以內(nèi)容執(zhí)行時間過長適當加大鎖的時間
    String key() default "" ;  //唯一標識,如果沒有那么默認為token->sessionId
    String doc() default "重復(fù)提交請求,請稍后再試";
    boolean repeatLock() default false; //可重復(fù)加鎖直到加鎖成功,默認為false不能重復(fù)加鎖
    int repeatLockCount() default -1; //可重復(fù)加鎖限制加鎖的次數(shù), 默認-1直到成功,設(shè)置10那么加鎖10次都沒成功就直接返回
    int lockWaitTimeMs() default 100; //重復(fù)加鎖默認的阻塞時間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 {
                // 執(zhí)行
                result = pjp.proceed();
            } finally {
                // 解鎖
                redisLock.release(key);
            }
            return result;
        } else {
            // 獲取鎖失敗,認為是重復(fù)提交的請求
            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<>();//計數(shù)

    //加鎖 , 線程加鎖的時候發(fā)現(xiàn),鎖有人用了 ,那么就會進入自旋等待
    @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());
            //獲取加鎖的次數(shù),如果是-1那么持續(xù)加鎖,如果滿足加鎖次數(shù)那么結(jié)束加鎖
            if (repeatLockCount != -1 && count.get().equals(repeatLockCount)) {
                logger.warn(Thread.currentThread().getName() + "嘗試加鎖失敗以嘗試了:" + count.get() + "次");
                return false;
            }
            SleepTools.ms(lockWaitTimeMs);
            //加鎖次數(shù)增加
            count.set(count.get() + 1);
        } while (repeatLock);
        logger.warn("重復(fù)提交");
        return false;
    }

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

    }
}

功能使用和介紹

  • 支持配置鎖的時間
  • 支持配置鎖的key(同一個key的請求會被鎖住)
  • 支持重復(fù)加鎖
  • 支持重復(fù)加鎖的次數(shù)
  • 支持重復(fù)加鎖的間隔時間

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

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

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

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

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

其他悲觀鎖的實現(xiàn)方式

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

        //獲取鎖對象
        RLock mylock = redissonClient.getLock(key);
        //一直等待直到加鎖成功后,并且設(shè)置鎖過期時間,防止死鎖的產(chǎn)生
        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是否上鎖,不然業(yè)務(wù)執(zhí)行時間大于鎖自動釋放時間后,解鎖報異常
        if(mylock.isLocked()){ // 是否還是鎖定狀態(tài)
            if(mylock.isHeldByCurrentThread()){ // 時候是當前執(zhí)行線程的鎖
                mylock.unlock(); // 釋放鎖
                System.err.println("======unlock======"+Thread.currentThread().getName());
            }
        }
    }
}

到此這篇關(guān)于Java-Redis-Redisson-分布式鎖的文章就介紹到這了,更多相關(guān)Java-Redis-Redisson分布式鎖內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • java迭代器基礎(chǔ)知識點總結(jié)

    java迭代器基礎(chǔ)知識點總結(jié)

    在本篇內(nèi)容里小編給大家整理了一篇關(guān)于java迭代器基礎(chǔ)知識點總結(jié)內(nèi)容,有興趣的朋友們可以學(xué)習參考下。
    2021-01-01
  • SpringSecurity框架下實現(xiàn)CSRF跨站攻擊防御的方法

    SpringSecurity框架下實現(xiàn)CSRF跨站攻擊防御的方法

    CSRF是一種網(wǎng)絡(luò)攻擊方式,也可以說是一種安全漏洞,這種安全漏洞在web開發(fā)中廣泛存在。這篇文章主要介紹了SpringSecurity框架下實現(xiàn)CSRF跨站攻擊防御,需要的朋友可以參考下
    2019-12-12
  • Java中int和Integer的區(qū)別

    Java中int和Integer的區(qū)別

    這篇文章主要介紹的是?Java中int和Integer的區(qū)別,Java?是一種強數(shù)據(jù)類型的語言,因此所有的屬性必須有一個數(shù)據(jù)類型,下面文章基于Java詳細int和Integer有何區(qū)別,需要的朋友可以參考一下
    2021-11-11
  • mybatis批量新增、刪除、查詢和修改方式

    mybatis批量新增、刪除、查詢和修改方式

    這篇文章主要介紹了mybatis批量新增、刪除、查詢和修改方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2021-09-09
  • Java8方法引用和構(gòu)造引用代碼實例

    Java8方法引用和構(gòu)造引用代碼實例

    這篇文章主要介紹了java8方法引用和構(gòu)造引用代碼實例,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習或者工作具有一定的參考學(xué)習價值,需要的朋友可以參考下
    2019-12-12
  • 深入解析Java編程中的StringBuffer與StringBuider

    深入解析Java編程中的StringBuffer與StringBuider

    這篇文章主要介紹了Java編程中的StringBuffer與StringBuider,是Java入門學(xué)習中的基礎(chǔ)知識,需要的朋友可以參考下
    2015-09-09
  • 淺談SSH框架中spring的原理

    淺談SSH框架中spring的原理

    下面小編就為大家?guī)硪黄獪\談SSH框架中spring的原理。小編覺得挺不錯的,現(xiàn)在就分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2017-01-01
  • JDBC工具類實現(xiàn)登錄功能

    JDBC工具類實現(xiàn)登錄功能

    這篇文章主要為大家詳細介紹了JDBC工具類實現(xiàn)登錄功能,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2021-07-07
  • SpringBoot @PostConstruct原理用法解析

    SpringBoot @PostConstruct原理用法解析

    這篇文章主要介紹了SpringBoot @PostConstruct原理用法解析,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習或者工作具有一定的參考學(xué)習價值,需要的朋友可以參考下
    2020-08-08
  • Java如何實現(xiàn)雙向鏈表功能

    Java如何實現(xiàn)雙向鏈表功能

    雙向鏈表也叫雙鏈表,是鏈表的一種,它的每個數(shù)據(jù)結(jié)點中都有兩個指針,分別指向直接后繼和直接前驅(qū)。所以,從雙向鏈表中的任意一個結(jié)點開始,都可以很方便地訪問它的前驅(qū)結(jié)點和后繼結(jié)點。一般我們都構(gòu)造雙向循環(huán)鏈表
    2021-11-11

最新評論