电脑知识|欧美黑人一区二区三区|软件|欧美黑人一级爽快片淫片高清|系统|欧美黑人狂野猛交老妇|数据库|服务器|编程开发|网络运营|知识问答|技术教程文章 - 好吧啦网

您的位置:首頁技術(shù)文章
文章詳情頁

Java redisson實現(xiàn)分布式鎖原理詳解

瀏覽:85日期:2022-09-05 09:54:06

Redisson分布式鎖

之前的基于注解的鎖有一種鎖是基本redis的分布式鎖,鎖的實現(xiàn)我是基于redisson組件提供的RLock,這篇來看看redisson是如何實現(xiàn)鎖的。

不同版本實現(xiàn)鎖的機制并不相同

引用的redisson最近發(fā)布的版本3.2.3,不同的版本可能實現(xiàn)鎖的機制并不相同,早期版本好像是采用簡單的setnx,getset等常規(guī)命令來配置完成,而后期由于redis支持了腳本Lua變更了實現(xiàn)原理。

<dependency><groupId>org.redisson</groupId><artifactId>redisson</artifactId><version>3.2.3</version></dependency>

setnx需要配合getset以及事務(wù)來完成,這樣才能比較好的避免死鎖問題,而新版本由于支持lua腳本,可以避免使用事務(wù)以及操作多個redis命令,語義表達更加清晰一些。

RLock接口的特點

繼承標(biāo)準(zhǔn)接口Lock

擁有標(biāo)準(zhǔn)鎖接口的所有特性,比如lock,unlock,trylock等等。

擴展標(biāo)準(zhǔn)接口Lock

展了很多方法,常用的主要有:強制鎖釋放,帶有效期的鎖,還有一組異步的方法。其中前面兩個方法主要是解決標(biāo)準(zhǔn)lock可能造成的死鎖問題。比如某個線程獲取到鎖之后,線程所在機器死機,此時獲取了鎖的線程無法正常釋放鎖導(dǎo)致其余的等待鎖的線程一直等待下去。

可重入機制

各版本實現(xiàn)有差異,可重入主要考慮的是性能,同一線程在未釋放鎖時如果再次申請鎖資源不需要走申請流程,只需要將已經(jīng)獲取的鎖繼續(xù)返回并且記錄上已經(jīng)重入的次數(shù)即可,與jdk里面的ReentrantLock功能類似。重入次數(shù)靠hincrby命令來配合使用,詳細(xì)的參數(shù)下面的代碼。

怎么判斷是同一線程?

redisson的方案是,RedissonLock實例的一個guid再加當(dāng)前線程的id,通過getLockName返回

public class RedissonLock extends RedissonExpirable implements RLock { final UUID id; protected RedissonLock(CommandExecutor commandExecutor, String name, UUID id) { super(commandExecutor, name); this.internalLockLeaseTime = TimeUnit.SECONDS.toMillis(30L); this.commandExecutor = commandExecutor; this.id = id; } String getLockName(long threadId) { return this.id + ':' + threadId; }

RLock獲取鎖的兩種場景

這里拿tryLock的源碼來看:tryAcquire方法是申請鎖并返回鎖有效期還剩余的時間,如果為空說明鎖未被其它線程申請直接獲取并返回,如果獲取到時間,則進入等待競爭邏輯。

public boolean tryLock(long waitTime, long leaseTime, TimeUnit unit) throws InterruptedException { long time = unit.toMillis(waitTime); long current = System.currentTimeMillis(); final long threadId = Thread.currentThread().getId(); Long ttl = this.tryAcquire(leaseTime, unit); if(ttl == null) { //直接獲取到鎖 return true; } else { //有競爭的后續(xù)看 } }

無競爭,直接獲取鎖

先看下首先獲取鎖并釋放鎖背后的redis都在做什么,可以利用redis的monitor來在后臺監(jiān)控redis的執(zhí)行情況。當(dāng)我們在方法了增加@RequestLockable之后,其實就是調(diào)用lock以及unlock,下面是redis命令:

加鎖

由于高版本的redis支持lua腳本,所以redisson也對其進行了支持,采用了腳本模式,不熟悉lua腳本的可以去查找下。執(zhí)行l(wèi)ua命令的邏輯如下:

<T> RFuture<T> tryLockInnerAsync(long leaseTime, TimeUnit unit, long threadId, RedisStrictCommand<T> command) { this.internalLockLeaseTime = unit.toMillis(leaseTime); return this.commandExecutor.evalWriteAsync(this.getName(), LongCodec.INSTANCE, command, 'if (redis.call(’exists’, KEYS[1]) == 0) then redis.call(’hset’, KEYS[1], ARGV[2], 1); redis.call(’pexpire’, KEYS[1], ARGV[1]); return nil; end; if (redis.call(’hexists’, KEYS[1], ARGV[2]) == 1) then redis.call(’hincrby’, KEYS[1], ARGV[2], 1); redis.call(’pexpire’, KEYS[1], ARGV[1]); return nil; end; return redis.call(’pttl’, KEYS[1]);', Collections.singletonList(this.getName()), new Object[]{Long.valueOf(this.internalLockLeaseTime), this.getLockName(threadId)}); }

加鎖的流程:

判斷l(xiāng)ock鍵是否存在,不存在直接調(diào)用hset存儲當(dāng)前線程信息并且設(shè)置過期時間,返回nil,告訴客戶端直接獲取到鎖。 判斷l(xiāng)ock鍵是否存在,存在則將重入次數(shù)加1,并重新設(shè)置過期時間,返回nil,告訴客戶端直接獲取到鎖。 被其它線程已經(jīng)鎖定,返回鎖有效期的剩余時間,告訴客戶端需要等待。

'EVAL' 'if (redis.call(’exists’, KEYS[1]) == 0) then redis.call(’hset’, KEYS[1], ARGV[2], 1); redis.call(’pexpire’, KEYS[1], ARGV[1]); return nil; end; if (redis.call(’hexists’, KEYS[1], ARGV[2]) == 1) then redis.call(’hincrby’, KEYS[1], ARGV[2], 1); redis.call(’pexpire’, KEYS[1], ARGV[1]); return nil; end; return redis.call(’pttl’, KEYS[1]);' '1' 'lock.com.csp.product.api.service.ProductAppService.searchProductByPage#0' '1000' '346e1eb8-5bfd-4d49-9870-042df402f248:21'

上面的lua腳本會轉(zhuǎn)換成真正的redis命令,下面的是經(jīng)過lua腳本運算之后實際執(zhí)行的redis命令。

1486642677.053488 [0 lua] 'exists' 'lock.com.csp.product.api.service.ProductAppService.searchProductByPage#0'1486642677.053515 [0 lua] 'hset' 'lock.com.csp.product.api.service.ProductAppService.searchProductByPage#0' '346e1eb8-5bfd-4d49-9870-042df402f248:21' '1'1486642677.053540 [0 lua] 'pexpire' 'lock.com.csp.product.api.service.ProductAppService.searchProductByPage#0' '1000'

解鎖

解鎖的流程看起來復(fù)雜些:

如果lock鍵不存在,發(fā)消息說鎖已經(jīng)可用 如果鎖不是被當(dāng)前線程鎖定,則返回nil 由于支持可重入,在解鎖時將重入次數(shù)需要減1 如果計算后的重入次數(shù)>0,則重新設(shè)置過期時間 如果計算后的重入次數(shù)<=0,則發(fā)消息說鎖已經(jīng)可用

'EVAL' 'if (redis.call(’exists’, KEYS[1]) == 0) then redis.call(’publish’, KEYS[2], ARGV[1]); return 1; end;if (redis.call(’hexists’, KEYS[1], ARGV[3]) == 0) then return nil;end; local counter = redis.call(’hincrby’, KEYS[1], ARGV[3], -1); if (counter > 0) then redis.call(’pexpire’, KEYS[1], ARGV[2]); return 0; else redis.call(’del’, KEYS[1]); redis.call(’publish’, KEYS[2], ARGV[1]); return 1; end; return nil;''2' 'lock.com.csp.product.api.service.ProductAppService.searchProductByPage#0' 'redisson_lock__channel:{lock.com.csp.product.api.service.ProductAppService.searchProductByPage#0}' '0' '1000' '346e1eb8-5bfd-4d49-9870-042df402f248:21'

無競爭情況下解鎖redis命令:主要是發(fā)送一個解鎖的消息,以此喚醒等待隊列中的線程重新競爭鎖。

1486642678.493691 [0 lua] 'exists' 'lock.com.csp.product.api.service.ProductAppService.searchProductByPage#0'1486642678.493712 [0 lua] 'publish' 'redisson_lock__channel:{lock.com.csp.product.api.service.ProductAppService.searchProductByPage#0}' '0'

有競爭,等待

有競爭的情況在redis端的lua腳本是相同的,只是不同的條件執(zhí)行不同的redis命令,復(fù)雜的在redisson的源碼上。當(dāng)通過tryAcquire發(fā)現(xiàn)鎖被其它線程申請時,需要進入等待競爭邏輯中。

this.await返回false,說明等待時間已經(jīng)超出獲取鎖最大等待時間,取消訂閱并返回獲取鎖失敗 this.await返回true,進入循環(huán)嘗試獲取鎖。

public boolean tryLock(long waitTime, long leaseTime, TimeUnit unit) throws InterruptedException { long time = unit.toMillis(waitTime); long current = System.currentTimeMillis(); final long threadId = Thread.currentThread().getId(); Long ttl = this.tryAcquire(leaseTime, unit); if(ttl == null) { return true; } else { //重點是這段 time -= System.currentTimeMillis() - current; if(time <= 0L) {return false; } else {current = System.currentTimeMillis();final RFuture subscribeFuture = this.subscribe(threadId);if(!this.await(subscribeFuture, time, TimeUnit.MILLISECONDS)) { if(!subscribeFuture.cancel(false)) { subscribeFuture.addListener(new FutureListener() { public void operationComplete(Future<RedissonLockEntry> future) throws Exception {if(subscribeFuture.isSuccess()) { RedissonLock.this.unsubscribe(subscribeFuture, threadId);} } }); } return false;} else { boolean var16; try { time -= System.currentTimeMillis() - current; if(time <= 0L) { boolean currentTime1 = false; return currentTime1; } do { long currentTime = System.currentTimeMillis(); ttl = this.tryAcquire(leaseTime, unit); if(ttl == null) {var16 = true;return var16; } time -= System.currentTimeMillis() - currentTime; if(time <= 0L) {var16 = false;return var16; } currentTime = System.currentTimeMillis(); if(ttl.longValue() >= 0L && ttl.longValue() < time) {this.getEntry(threadId).getLatch().tryAcquire(ttl.longValue(), TimeUnit.MILLISECONDS); } else {this.getEntry(threadId).getLatch().tryAcquire(time, TimeUnit.MILLISECONDS); } time -= System.currentTimeMillis() - currentTime; } while(time > 0L); var16 = false; } finally { this.unsubscribe(subscribeFuture, threadId); } return var16;} } } }

循環(huán)嘗試一般有如下幾種方法:

while循環(huán),一次接著一次的嘗試,這個方法的缺點是會造成大量無效的鎖申請。 Thread.sleep,在上面的while方案中增加睡眠時間以降低鎖申請次數(shù),缺點是這個睡眠的時間設(shè)置比較難控制。 基于信息量,當(dāng)鎖被其它資源占用時,當(dāng)前線程訂閱鎖的釋放事件,一旦鎖釋放會發(fā)消息通知待等待的鎖進行競爭,有效的解決了無效的鎖申請情況。核心邏輯是this.getEntry(threadId).getLatch().tryAcquire,this.getEntry(threadId).getLatch()返回的是一個信號量,有興趣可以再研究研究。

redisson依賴

由于redisson不光是針對鎖,提供了很多客戶端操作redis的方法,所以會依賴一些其它的框架,比如netty,如果只是簡單的使用鎖也可以自己去實現(xiàn)。以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持好吧啦網(wǎng)。

標(biāo)簽: Java
相關(guān)文章:
主站蜘蛛池模板: 成都中天自动化控制技术有限公司| 防火门-专业生产甲级不锈钢钢质防火门厂家资质齐全-广东恒磊安防设备有限公司 | 不锈钢法兰-碳钢法兰-法兰盘生产加工厂家-[鼎捷峰]-不锈钢法兰-碳钢法兰-法兰盘生产加工厂家-[鼎捷峰] | 河南空气能热水器-洛阳空气能采暖-洛阳太阳能热水工程-洛阳润达高科空气能商行 | 首页|光催化反应器_平行反应仪_光化学反应仪-北京普林塞斯科技有限公司 | 浙江红酒库-冰雕库-气调库-茶叶库安装-医药疫苗冷库-食品物流恒温恒湿车间-杭州领顺实业有限公司 | 定制奶茶纸杯_定制豆浆杯_广东纸杯厂_[绿保佳]一家专业生产纸杯碗的厂家 | 管形母线,全绝缘铜管母线厂家-山东佰特电气科技有限公司 | 环氧铁红防锈漆_环氧漆_无溶剂环氧涂料_环氧防腐漆-华川涂料 | 温州中研白癜风专科_温州治疗白癜风_温州治疗白癜风医院哪家好_温州哪里治疗白癜风 | 免费B2B信息推广发布平台 - 推发网| 酒精检测棒,数显温湿度计,酒安酒精测试仪,酒精检测仪,呼气式酒精检测仪-郑州欧诺仪器有限公司 | 青岛侦探调查_青岛侦探事务所_青岛调查事务所_青岛婚外情取证-青岛狄仁杰国际侦探公司 | 换网器_自动换网器_液压换网器--郑州海科熔体泵有限公司 | 幂简集成 - 品种超全的API接口平台, 一站搜索、试用、集成国内外API接口 | 浙江上沪阀门有限公司 | 广东青藤环境科技有限公司-水质检测| 广东燎了网络科技有限公司官网-网站建设-珠海网络推广-高端营销型外贸网站建设-珠海专业h5建站公司「了了网」 | 镀锌钢格栅_热镀锌格栅板_钢格栅板_热镀锌钢格板-安平县昊泽丝网制品有限公司 | 急救箱-应急箱-急救包厂家-北京红立方医疗设备有限公司 | 今日扫码_溯源二维码_产品防伪一物一码_红包墙营销方案 | 聚合氯化铝-碱式氯化铝-聚合硫酸铁-聚氯化铝铁生产厂家多少钱一吨-聚丙烯酰胺价格_河南浩博净水材料有限公司 | 东莞工厂厂房装修_无尘车间施工_钢结构工程安装-广东集景建筑装饰设计工程有限公司 | 膏剂灌装旋盖机-眼药水灌装生产线-西林瓶粉剂分装机-南通博琅机械科技 | 生鲜配送系统-蔬菜食材配送管理系统-连锁餐饮订货配送软件-挪挪生鲜供应链管理软件 | 氧化锆纤维_1800度高温退火炉_1800度高温烧结炉-南京理工宇龙新材料股份有限公司 | 仓储笼_金属箱租赁_循环包装_铁网箱_蝴蝶笼租赁_酷龙仓储笼租赁 测试治具|过炉治具|过锡炉治具|工装夹具|测试夹具|允睿自动化设备 | 铝合金重力铸造_铝合金翻砂铸造_铝铸件厂家-东莞市铝得旺五金制品有限公司 | 卫生纸复卷机|抽纸机|卫生纸加工设备|做卫生纸机器|小型卫生纸加工需要什么设备|卫生纸机器设备多少钱一台|许昌恒源纸品机械有限公司 | 磁力去毛刺机_去毛刺磁力抛光机_磁力光饰机_磁力滚抛机_精密金属零件去毛刺机厂家-冠古科技 | CCC验厂-家用电器|服务器CCC认证咨询-奥测世纪 | 软启动器-上海能曼电气有限公司 真空搅拌机-行星搅拌机-双行星动力混合机-广州市番禺区源创化工设备厂 | 申江储气罐厂家,储气罐批发价格,储气罐规格-上海申江压力容器有限公司(厂) | 喷漆房_废气处理设备-湖北天地鑫环保设备有限公司 | 无线联网门锁|校园联网门锁|学校智能门锁|公租房智能门锁|保障房管理系统-KEENZY中科易安 | 智慧物联网行业一站式解决方案提供商-北京东成基业 | 杭州代理记账多少钱-注册公司代办-公司注销流程及费用-杭州福道财务管理咨询有限公司 | 校园气象站_超声波气象站_农业气象站_雨量监测站_风途科技 | 长沙印刷厂-包装印刷-画册印刷厂家-湖南省日大彩色印务有限公司 青州搬家公司电话_青州搬家公司哪家好「鸿喜」青州搬家 | 3D全息投影_地面互动投影_360度立体投影_水幕灯光秀 | 广东银虎 蜂窝块状沸石分子筛-吸附脱硫分子筛-萍乡市捷龙环保科技有限公司 | 专业深孔加工_东莞深孔钻加工_东莞深孔钻_东莞深孔加工_模具深孔钻加工厂-东莞市超耀实业有限公司 |