編輯器軟體LuaStudio
一:Lua腳本
說來也巧,redis的大老闆給了你解決這種問題的方法,那就是Lua腳本,而且redis的最新版本也支援Lua Script debug,這應該也是未來Redis的一
個發展趨勢,要想學好Redis,必會Lua Script。。。
作為一門過程型動态語言,Lua有着如下的特性
1、變量名m沒有類型,值才有類型,變量名在運作時可與任何類型的值綁定。
2、語言隻提供唯一一種資料結構,稱為表(Table),它混合了數組、hash。可以使用任何類型的值作為key和value,提供了一緻且富有表達力的表構造文法,使得Lua很适合描述複雜h和資料。
3、函數s是一等類型,支援匿名函數和正則w尾遞歸
4、支援詞法定界、和閉包。
5、提供thread類型和結構化的協程(coroutine)機制,在此基礎上可友善實作協作式多任務。
6、運作期能b編譯字元串形式的程式文本并載入虛拟機執行。
7、通過元表和元方法提供動态元機制,進而允許程式運作時根據需要改變或擴充文法設施的内定語義。
8、能友善利用表和動态元機制實作基于原型的面向對象模型。
有趣的是,官網上還提供了一個視訊教程教你如何進行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中給大家示範一下:
[[email protected] 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放到一個lua檔案中,然後執行這個lua腳本,比如
下面這樣:
然後我們通過下面指令執行,這種方式和前面介紹的不一樣,參數 --eval script key1 key2 , arg1 age2 這種模式,key和value用一個逗号隔開就好了,
最後我們也看到了,資料都出來了,對吧。
[[email protected] Desktop]# redis-cli --eval /usr/redis/sbin/1.lua username age , jack 20
1) "username"
2) "age"
3) "jack"
4) "20"
[[email protected] Desktop]#
三:實戰
下面我可以構思幾個小案例通過lua解決。
1. 通過lua腳本擷取指定的key的List中的所有資料
local key=KEYS[1]
local list=redis.call("lrange",key,0,-1);
return list;
這裡面的redis.call就是用來執行redis中list的lrange指令,接下來我通過lpush給person塞入三條資料,如下:
[[email protected] 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",key,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,1,v);
end
end
return result;