1. 程式人生 > 資料庫 >redis中如何使用lua指令碼讓你的靈活性提高5個逼格詳解

redis中如何使用lua指令碼讓你的靈活性提高5個逼格詳解

前言

在實際工作過程中,可以使用lua指令碼來解決一些需要保證原子性的問題,而且lua指令碼可以快取在redis伺服器上,勢必會增加效能。

然而在redis的官網上洋洋灑灑的大概提供了200多個命令,貌似看起來很多,但是這些都是別人預先給你定義好的,但你卻不能按照自己的意圖進行定製,

所以是不是感覺自己還是有一種被束縛的感覺,有這個感覺就對了。。。

一:Lua指令碼

說來也巧,redis的大老闆給了你解決這種問題的方法,那就是Lua指令碼,而且redis的最新版本也支援Lua Script debug,這應該也是未來Redis的一

個發展趨勢,要想學好Redis,必會Lua Script。。。

有趣的是,官網上還提供了一個視訊教程教你如何進行Debug操作。。。 【https://redis.io/topics/ldb】 youtube上面的視訊,要是被牆了,記得

上VPN哦。。。淘寶上不知道有沒有售賣這種同款的吸頂燈~~~

二:使用Redis-Cli Lua Script 解決幾個靈活性問題

1. Lua語法的問題

lua是一門程式語言,所以這個就已經超出了redis本身的範疇,如果大家想好好學習一下,可以看下http://www.lua.org/ 的官網,然後下載一下玩一玩。

比如這裡我下載了一個windows版本的lua 編譯器,具體語法上就不細說了。。有了這個主題,我們再進行下一個環節。

2. Eval的使用

EVAL script numkeys key [key ...] arg [arg ...]

首先大家一定要知道eval的語法格式,其中:

<1> script: 你的lua指令碼

<2> numkeys: key的個數

<3> key: redis中各種資料結構的替代符號

<4> arg: 你的自定義引數

ok,可能乍一看模板不是特別清楚,下面我可以用官網的小案例演示一下:

eval "return {KEYS[1],KEYS[2],ARGV[1],ARGV[2]}" 2 username age jack 20

上面這一串程式碼大概是什麼意思呢? 第一個引數的字串就是script,也就是lua指令碼。2表示keys的個數,KEYS[1] 就是 username的佔位符,KEYS[2]就是

age的佔位符,ARGV[1]就是jack的佔位符,ARGV[2]就是20的佔位符,,以此類推,,,所以最後的結果應該就是:{return username age jack 20} 是不

是有點像C#中的佔位符:{0}呢???下面我在Redis中給大家演示一下:

[root@localhost Desktop]# redis-cli
127.0.0.1:6379> eval "return {KEYS[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放到一個lua檔案中,然後執行這個lua指令碼,比如

下面這樣:

然後我們通過下面命令執行,這種方式和前面介紹的不一樣,引數 --eval script key1 key2,arg1 age2 這種模式,key和value用一個逗號隔開就好了,

最後我們也看到了,資料都出來了,對吧。

[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]# 

三:實戰

下面我可以構思幾個小案例通過lua解決。

1. 通過lua指令碼獲取指定的key的List中的所有資料

local key=KEYS[1]

local list=redis.call("lrange",key,-1);

return list;

這裡面的redis.call就是用來執行redis中list的lrange命令,接下來我通過lpush給person塞入三條資料,如下:

[root@localhost Desktop]# redis-cli
127.0.0.1:6379> lpush person mary jack peter
(integer) 3
127.0.0.1:6379> 

然後我們來執行這個lua指令碼,效果如下圖,是不是很牛逼的感覺???

有了這個1+1的效果,就可以玩些更復雜的操作。比如:

2.根據外面傳過來的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",n);

 if(ishit) then
  table.insert(result,1,n);
 end

 end

return result;

2. 找到hash中age小於指定值的所有資料,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,v);
 end
end

return result;

大家可以試著看下這段指令碼,然後按照這個邏輯自己玩一玩,很有意思的,還是那句話,學好redis,必會Lua。。。。

總結

以上就是這篇文章的全部內容了,希望本文的內容對大家的學習或者工作具有一定的參考學習價值,如果有疑問大家可以留言交流,謝謝大家對我們的支援。