前言
在實(shí)際工作過程中,可以使用lua腳本來解決一些需要保證原子性的問題,而且lua腳本可以緩存在redis服務(wù)器上,勢必會(huì)增加性能。
然而在redis的官網(wǎng)上洋洋灑灑的大概提供了200多個(gè)命令,貌似看起來很多,但是這些都是別人預(yù)先給你定義好的,但你卻不能按照自己的意圖進(jìn)行定制,
所以是不是感覺自己還是有一種被束縛的感覺,有這個(gè)感覺就對了。。。
一:Lua腳本
說來也巧,redis的大老板給了你解決這種問題的方法,那就是Lua腳本,而且redis的最新版本也支持Lua Script debug,這應(yīng)該也是未來Redis的一
個(gè)發(fā)展趨勢,要想學(xué)好Redis,必會(huì)Lua Script。。。
![](/d/20211018/ec20219028946c2d10d10d9b33603579.gif)
有趣的是,官網(wǎng)上還提供了一個(gè)視頻教程教你如何進(jìn)行Debug操作。。。 【https://redis.io/topics/ldb】 youtube上面的視頻,要是被墻了,記得
上VPN哦。。。淘寶上不知道有沒有售賣這種同款的吸頂燈~~~
![](/d/20211018/6650198270c2abbebeb0801b3c3be83c.gif)
二:使用Redis-Cli Lua Script 解決幾個(gè)靈活性問題
1. Lua語法的問題
lua是一門編程語言,所以這個(gè)就已經(jīng)超出了redis本身的范疇,如果大家想好好學(xué)習(xí)一下,可以看下http://www.lua.org/ 的官網(wǎng),然后下載一下玩一玩。
![](/d/20211018/405921849053578ba723a854a2c6072a.gif)
比如這里我下載了一個(gè)windows版本的lua 編譯器,具體語法上就不細(xì)說了。。有了這個(gè)主題,我們再進(jìn)行下一個(gè)環(huán)節(jié)。
![](/d/20211018/1476c63ed3506d993f478806c2fdfeed.gif)
2. Eval的使用
EVAL script numkeys key [key ...] arg [arg ...]
首先大家一定要知道eval的語法格式,其中:
1> script: 你的lua腳本
2> numkeys: key的個(gè)數(shù)
3> key: redis中各種數(shù)據(jù)結(jié)構(gòu)的替代符號
4> arg: 你的自定義參數(shù)
ok,可能乍一看模板不是特別清楚,下面我可以用官網(wǎng)的小案例演示一下:
eval "return {KEYS[1],KEYS[2],ARGV[1],ARGV[2]}" 2 username age jack 20
上面這一串代碼大概是什么意思呢? 第一個(gè)參數(shù)的字符串就是script,也就是lua腳本。2表示keys的個(gè)數(shù),KEYS[1] 就是 username的占位符, KEYS[2]就是
age的占位符,ARGV[1]就是jack的占位符,ARGV[2]就是20的占位符,,以此類推,,,所以最后的結(jié)果應(yīng)該就是:{return username age jack 20} 是不
是有點(diǎn)像C#中的占位符:{0}呢???下面我在Redis中給大家演示一下:
[root@localhost Desktop]# redis-cli
127.0.0.1:6379> eval "return {KEYS[1],KEYS[2],ARGV[1],ARGV[2]}" 2 username age jack 20
1) "username"
2) "age"
3) "jack"
4) "20"
127.0.0.1:6379>
通常境況下,我們不要在redis-cli中直接寫lua腳本,這樣非常不方便編輯,通常情況下我們都是把lua script放到一個(gè)lua文件中,然后執(zhí)行這個(gè)lua腳本,比如
下面這樣:
![](/d/20211018/ff1564da368def2efd99a400722186c2.gif)
然后我們通過下面命令執(zhí)行,這種方式和前面介紹的不一樣,參數(shù) --eval script key1 key2 , arg1 age2 這種模式,key和value用一個(gè)逗號隔開就好了,
最后我們也看到了,數(shù)據(jù)都出來了,對吧。
[root@localhost Desktop]# redis-cli --eval /usr/redis/sbin/1.lua username age , jack 20
1) "username"
2) "age"
3) "jack"
4) "20"
[root@localhost Desktop]#
三:實(shí)戰(zhàn)
下面我可以構(gòu)思幾個(gè)小案例通過lua解決。
1. 通過lua腳本獲取指定的key的List中的所有數(shù)據(jù)
local key=KEYS[1]
local list=redis.call("lrange",key,0,-1);
return list;
這里面的redis.call就是用來執(zhí)行redis中l(wèi)ist的lrange命令,接下來我通過lpush給person塞入三條數(shù)據(jù),如下:
[root@localhost Desktop]# redis-cli
127.0.0.1:6379> lpush person mary jack peter
(integer) 3
127.0.0.1:6379>
然后我們來執(zhí)行這個(gè)lua腳本,效果如下圖,是不是很牛逼的感覺???
![](/d/20211018/9e652b52bc76af3e612948156b225924.gif)
有了這個(gè)1+1的效果,就可以玩些更復(fù)雜的操作。比如:
2.根據(jù)外面?zhèn)鬟^來的IDList 做“集合去重”的lua腳本邏輯:
local key=KEYS[1];
local args=ARGV
local i=0;
local result={};
for m,n in ipairs(args) do
local ishit=redis.call("sismember",key,n);
if(ishit) then
table.insert(result,1,n);
end
end
return result;
2. 找到hash中age小于指定值的所有數(shù)據(jù),lua腳本如下:
local result={};
local myperson=KEYS[1];
local nums=ARGV[1];
local myresult =redis.call("hkeys",myperson);
for i,v in ipairs(myresult) do
local hval= redis.call("hget",myperson,v);
redis.log(redis.LOG_WARNING,hval);
if(tonumber(hval)tonumber(nums)) then
table.insert(result,1,v);
end
end
return result;
大家可以試著看下這段腳本,然后按照這個(gè)邏輯自己玩一玩,很有意思的,還是那句話,學(xué)好redis,必會(huì)Lua。。。。
總結(jié)
以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,如果有疑問大家可以留言交流,謝謝大家對腳本之家的支持。
您可能感興趣的文章:
-
SpringBoot通過RedisTemplate執(zhí)行Lua腳本的方法步驟
-
Redis執(zhí)行Lua腳本的好處與示例代碼
-
Go語言中通過Lua腳本操作Redis的方法
-
Redis分布式鎖的實(shí)現(xiàn)方式(redis面試題)
-
SpringBoot使用Redisson實(shí)現(xiàn)分布式鎖(秒殺系統(tǒng))
-
SpringBoot集成Redisson實(shí)現(xiàn)分布式鎖的方法示例
-
Java Redis分布式鎖的正確實(shí)現(xiàn)方式詳解
-
基于Redis實(shí)現(xiàn)分布式鎖的方法(lua腳本版)