<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
Cash Loan(一):Redis實現計數器防刷 中介紹了專案中應用redis來做計數器的實現過程,最近自己看了些關於Redis實現分散式鎖的程式碼後,發現在Redis分散式鎖中出現一個問題在這版計數器中同樣會出現,於是融入了Lua指令碼進行升級改造有了Redis+Lua版本。
如果set命令設定上,但是在設定失效時間時由於網路抖動等原因導致沒有設定成功,這時就會出現死計數器(類似死鎖);
Redis+Lua是一個很好的解決方案,使用指令碼使得set命令和expire命令一同達到Redis被執行且不會被幹擾,在很大程度上保證了原子操作;
為什麼說是很大程度上保證原子操作而不是完全保證?因為在Redis內部執行的時候出問題也有可能出現問題不過概率非常小;即使針對小概率事件也有相應的解決方案,比如解決死鎖一個思路值得參考:防止死鎖會將鎖的值存成一個時間戳,即使發生沒有將失效時間設定上在判斷是否上鎖時可以加上看看其中值距現在是否超過一個設定的時間,如果超過則將其刪除重新設定鎖。
1、Redis+Lua鎖的實現
package han.zhang.utils; import org.springframework.data.redis.core.StringRedisTemplate; import org.springframework.data.redis.core.script.DigestUtils; import org.springframework.data.redis.core.script.RedisScript; import java.util.Collections; import java.util.UUID; public class RedisLock { private static final LogUtils logger = LogUtils.getLogger(RedisLock.class); private final StringRedisTemplate stringRedisTemplate; private final String lockKey; private final String lockValue; private boolean locked = false; /** * 使用指令碼在redis伺服器執行這個邏輯可以在一定程度上保證此操作的原子性 * (即不會發生使用者端在執行setNX和expire命令之間,發生崩潰或失去與伺服器的連線導致expire沒有得到執行,發生永久死鎖) * <p> * 除非指令碼在redis伺服器執行時redis伺服器發生崩潰,不過此種情況鎖也會失效 */ private static final RedisScript<Boolean> SETNX_AND_EXPIRE_SCRIPT; static { StringBuilder sb = new StringBuilder(); sb.append("if (redis.call('setnx', KEYS[1], ARGV[1]) == 1) thenn"); sb.append("tredis.call('expire', KEYS[1], tonumber(ARGV[2]))n"); sb.append("treturn truen"); sb.append("elsen"); sb.append("treturn falsen"); sb.append("end"); SETNX_AND_EXPIRE_SCRIPT = new RedisScriptImpl<>(sb.toString(), Boolean.class); } private static final RedisScript<Boolean> DEL_IF_GET_EQUALS; sb.append("if (redis.call('get', KEYS[1]) == ARGV[1]) thenn"); sb.append("tredis.call('del', KEYS[1])n"); DEL_IF_GET_EQUALS = new RedisScriptImpl<>(sb.toString(), Boolean.class); public RedisLock(StringRedisTemplate stringRedisTemplate, String lockKey) { this.stringRedisTemplate = stringRedisTemplate; this.lockKey = lockKey; this.lockValue = UUID.randomUUID().toString() + "." + System.currentTimeMillis(); private boolean doTryLock(int lockSeconds) { if (locked) { throw new IllegalStateException("already locked!"); } locked = stringRedisTemplate.execute(SETNX_AND_EXPIRE_SCRIPT, Collections.singletonList(lockKey), lockValue, String.valueOf(lockSeconds)); return locked; * 嘗試獲得鎖,成功返回true,如果失敗立即返回false * * @param lockSeconds 加鎖的時間(秒),超過這個時間後鎖會自動釋放 public boolean tryLock(int lockSeconds) { try { return doTryLock(lockSeconds); } catch (Exception e) { logger.error("tryLock Error", e); return false; * 輪詢的方式去獲得鎖,成功返回true,超過輪詢次數或異常返回false * @param lockSeconds 加鎖的時間(秒),超過這個時間後鎖會自動釋放 * @param tryIntervalMillis 輪詢的時間間隔(毫秒) * @param maxTryCount 最大的輪詢次數 public boolean tryLock(final int lockSeconds, final long tryIntervalMillis, final int maxTryCount) { int tryCount = 0; while (true) { if (++tryCount >= maxTryCount) { // 獲取鎖超時 return false; } try { if (doTryLock(lockSeconds)) { return true; } } catch (Exception e) { logger.error("tryLock Error", e); Thread.sleep(tryIntervalMillis); } catch (InterruptedException e) { logger.error("tryLock interrupted", e); * 解鎖操作 public void unlock() { if (!locked) { throw new IllegalStateException("not locked yet!"); locked = false; // 忽略結果 stringRedisTemplate.execute(DEL_IF_GET_EQUALS, Collections.singletonList(lockKey), lockValue); private static class RedisScriptImpl<T> implements RedisScript<T> { private final String script; private final String sha1; private final Class<T> resultType; public RedisScriptImpl(String script, Class<T> resultType) { this.script = script; this.sha1 = DigestUtils.sha1DigestAsHex(script); this.resultType = resultType; @Override public String getSha1() { return sha1; public Class<T> getResultType() { return resultType; public String getScriptAsString() { return script; }
2、借鑑鎖實現Redis+Lua計數器
(1)工具類
package han.zhang.utils; import org.springframework.data.redis.core.StringRedisTemplate; import org.springframework.data.redis.core.script.DigestUtils; import org.springframework.data.redis.core.script.RedisScript; import java.util.Collections; public class CountUtil { private static final LogUtils logger = LogUtils.getLogger(CountUtil.class); private final StringRedisTemplate stringRedisTemplate; /** * 使用指令碼在redis伺服器執行這個邏輯可以在一定程度上保證此操作的原子性 * (即不會發生使用者端在執行setNX和expire命令之間,發生崩潰或失去與伺服器的連線導致expire沒有得到執行,發生永久死計數器) * <p> * 除非指令碼在redis伺服器執行時redis伺服器發生崩潰,不過此種情況計數器也會失效 */ private static final RedisScript<Boolean> SET_AND_EXPIRE_SCRIPT; static { StringBuilder sb = new StringBuilder(); sb.append("local visitTimes = redis.call('incr', KEYS[1])n"); sb.append("if (visitTimes == 1) thenn"); sb.append("tredis.call('expire', KEYS[1], tonumber(ARGV[1]))n"); sb.append("treturn falsen"); sb.append("elseif(visitTimes > tonumber(ARGV[2])) thenn"); sb.append("treturn truen"); sb.append("elsen"); sb.append("end"); SET_AND_EXPIRE_SCRIPT = new RedisScriptImpl<>(sb.toString(), Boolean.class); } public CountUtil(StringRedisTemplate stringRedisTemplate) { this.stringRedisTemplate = stringRedisTemplate; public boolean isOverMaxVisitTimes(String key, int seconds, int maxTimes) throws Exception { try { return stringRedisTemplate.execute(SET_AND_EXPIRE_SCRIPT, Collections.singletonList(key), String.valueOf(seconds), String.valueOf(maxTimes)); } catch (Exception e) { logger.error("RedisBusiness>>>isOverMaxVisitTimes; get visit times Exception; key:" + key + "result:" + e.getMessage()); throw new Exception("already Over MaxVisitTimes"); } private static class RedisScriptImpl<T> implements RedisScript<T> { private final String script; private final String sha1; private final Class<T> resultType; public RedisScriptImpl(String script, Class<T> resultType) { this.script = script; this.sha1 = DigestUtils.sha1DigestAsHex(script); this.resultType = resultType; @Override public String getSha1() { return sha1; public Class<T> getResultType() { return resultType; public String getScriptAsString() { return script; }
(2)呼叫測試程式碼
public void run(String... strings) { CountUtil countUtil = new CountUtil(SpringUtils.getStringRedisTemplate()); try { for (int i = 0; i < 10; i++) { boolean overMax = countUtil.isOverMaxVisitTimes("zhanghantest", 600, 2); if (overMax) { System.out.println("超過i:" + i + ":" + overMax); } else { System.out.println("沒超過i:" + i + ":" + overMax); } } } catch (Exception e) { logger.error("Exception {}", e.getMessage()); } }
(3)測試結果
1、用心去不斷的改造自己的程式;
2、用程式碼改變世界。
到此這篇關於Redis+Lua實現計數器介面防刷(升級版)的文章就介紹到這了,更多相關Redis計數器內容請搜尋it145.com以前的文章或繼續瀏覽下面的相關文章希望大家以後多多支援it145.com!
相關文章
<em>Mac</em>Book项目 2009年学校开始实施<em>Mac</em>Book项目,所有师生配备一本<em>Mac</em>Book,并同步更新了校园无线网络。学校每周进行电脑技术更新,每月发送技术支持资料,极大改变了教学及学习方式。因此2011
2021-06-01 09:32:01
综合看Anker超能充系列的性价比很高,并且与不仅和iPhone12/苹果<em>Mac</em>Book很配,而且适合多设备充电需求的日常使用或差旅场景,不管是安卓还是Switch同样也能用得上它,希望这次分享能给准备购入充电器的小伙伴们有所
2021-06-01 09:31:42
除了L4WUDU与吴亦凡已经多次共事,成为了明面上的厂牌成员,吴亦凡还曾带领20XXCLUB全队参加2020年的一场音乐节,这也是20XXCLUB首次全员合照,王嗣尧Turbo、陈彦希Regi、<em>Mac</em> Ova Seas、林渝植等人全部出场。然而让
2021-06-01 09:31:34
目前应用IPFS的机构:1 谷歌<em>浏览器</em>支持IPFS分布式协议 2 万维网 (历史档案博物馆)数据库 3 火狐<em>浏览器</em>支持 IPFS分布式协议 4 EOS 等数字货币数据存储 5 美国国会图书馆,历史资料永久保存在 IPFS 6 加
2021-06-01 09:31:24
开拓者的车机是兼容苹果和<em>安卓</em>,虽然我不怎么用,但确实兼顾了我家人的很多需求:副驾的门板还配有解锁开关,有的时候老婆开车,下车的时候偶尔会忘记解锁,我在副驾驶可以自己开门:第二排设计很好,不仅配置了一个很大的
2021-06-01 09:30:48
不仅是<em>安卓</em>手机,苹果手机的降价力度也是前所未有了,iPhone12也“跳水价”了,发布价是6799元,如今已经跌至5308元,降价幅度超过1400元,最新定价确认了。iPhone12是苹果首款5G手机,同时也是全球首款5nm芯片的智能机,它
2021-06-01 09:30:45