Redis使用Lua腳本命令詳解
Lua腳本
redis可以支持lua腳本,可以使用lua腳本來(lái)將幾個(gè)命令整合為一個(gè)整體來(lái)執(zhí)行,這樣可以使得多個(gè)命令原子操作,且可以減少網(wǎng)絡(luò)開(kāi)銷(xiāo)
Lua的數(shù)據(jù)類(lèi)型
Lua是一個(gè)動(dòng)態(tài)類(lèi)型的語(yǔ)言,一個(gè)變量可以存儲(chǔ)任何類(lèi)型的值,類(lèi)型有:
- 空:nil,也就是還沒(méi)有賦值
- 字符串:用單引號(hào) 或者 雙引號(hào)
- 數(shù)字:包含整數(shù)和浮點(diǎn)型
- 布爾:boolean
- 表:表是Lua唯一的數(shù)據(jù)結(jié)構(gòu),既可以當(dāng)數(shù)組,也可以做Map,或被視為對(duì)象
- 函數(shù):封裝某個(gè)或某些功能
- userData:用來(lái)將任意 C 數(shù)據(jù)保存在 Lua 變量中,這樣的操作只能通過(guò) C API
- Thread:用來(lái)區(qū)別獨(dú)立的執(zhí)行線程,它被用來(lái)實(shí)現(xiàn) coroutine (協(xié)同例程)
eval命令
redis內(nèi)有內(nèi)置的lua解釋器,可以使用eval命令對(duì)lua腳本進(jìn)行求值
# script是lua腳本 # numkeys指定鍵名參數(shù)的個(gè)數(shù) # key [key ...] 在腳本中使用的redis鍵,個(gè)數(shù)為numkeys指定的個(gè)數(shù),可以在lua中通過(guò)全局變量KEYS數(shù)組,從1開(kāi)始,KEYS[1],KEYS[2]等 # arg [arg ...] 參數(shù),可以在lua中通過(guò)全局變量ARGV數(shù)組訪問(wèn),ARGV[1],ARGV[2]等 eval script numkeys key [key ...] arg [arg ...] #示例: eval "return {KEYS[1],KEYS[2],ARGV[1]}" 2 key1 key2 first
還可以在lua腳本中調(diào)用redis命令,使用redis.call
eval "return redis.call('set',KEYS[1],'bar')" 1 foo
需要注意的是,redis執(zhí)行l(wèi)ua腳本和普通命令一樣,都是會(huì)寫(xiě)入AOF文件和發(fā)布至主從復(fù)制連接上的,有兩種方式
第一種方式:和普通命令相同,涉及到的寫(xiě)操作都會(huì)記錄/發(fā)送
普通命令會(huì)轉(zhuǎn)化為redis通信協(xié)議的格式,比起lua腳本來(lái)說(shuō),浪費(fèi)了更多的帶寬,而且slave接收到之后還需要再轉(zhuǎn)換為普通命令
第二種方式:直接將lua腳本記錄/發(fā)送
如果直接發(fā)送lua腳本,有些命令可能會(huì)導(dǎo)致每個(gè)機(jī)器執(zhí)行的結(jié)果不同,如取隨機(jī)數(shù)等
這個(gè)redis會(huì)決定采取哪種方式來(lái)執(zhí)行,在執(zhí)行前會(huì)進(jìn)行檢測(cè)寫(xiě)操作是否執(zhí)行了RANDOMKEY命令,來(lái)決定使用哪種方式
evalsha命令
考慮到腳本比較長(zhǎng)的情況,如果每次調(diào)用腳本都需要將整個(gè)腳本傳給redis會(huì)占用較多的帶寬。為了解決該問(wèn)題,redis提供了evalsha命令允許開(kāi)發(fā)者通過(guò)腳本內(nèi)容的SHA1摘要來(lái)執(zhí)行腳本,該命令的用法的eval一樣,不過(guò)是將腳本內(nèi)容替換為腳本內(nèi)容的SHA1摘要。
redis在執(zhí)行eval命令時(shí)會(huì)計(jì)算腳本的SHA1摘要并記錄在腳本緩存中,執(zhí)行evalsha命令時(shí)redis會(huì)根據(jù)提供的摘要從腳本緩存中查找對(duì)應(yīng)的腳本內(nèi)容,如果找到了則執(zhí)行腳本,否則返回錯(cuò)誤
使用流程
- 先計(jì)算腳本的SHA1摘要,并使用evalsha命令執(zhí)行腳本
- 獲的返回值,如果返回"NoScript"錯(cuò)誤則使用eval命令重新執(zhí)行腳本
其他不常用的命令
這些命令不是不常用,而是經(jīng)常被工具類(lèi)封裝起來(lái),開(kāi)發(fā)者如果不深入源碼查看,很少會(huì)用到
將腳本加入緩存
SCRIPT LOAD命令,作用是每次執(zhí)行eval命令時(shí)redis都會(huì)將腳本的SHA1摘要加入到腳本緩存中,以便下次客戶(hù)端可以使用evalsha命令調(diào)用該腳本。如果只是希望將腳本加入緩存而不執(zhí)行,則使用SCRIPT LOAD命令,返回值是腳本的SHA1摘要
判斷腳本是否被緩存
SCRIPT EXISTS命令,可以同時(shí)查找多個(gè)腳本的SHA1摘要是否被緩存
清空腳本緩存
SCRIPT FLUSH命令,redis將腳本的SHA1摘要加入到腳本緩存后會(huì)永久保留,不會(huì)刪除,可以使用該命令清空腳本緩存
強(qiáng)制終止當(dāng)前腳本的執(zhí)行
SCRIPT KILL命令,如果想終止當(dāng)前正在執(zhí)行的腳本可以使用該命令
以上就是Redis使用Lua腳本命令詳解的詳細(xì)內(nèi)容,更多關(guān)于Redis使用Lua腳本的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
- redis lua腳本實(shí)戰(zhàn)秒殺和減庫(kù)存的實(shí)現(xiàn)
- Redis中Lua腳本的使用和設(shè)置超時(shí)
- Redis+Lua腳本實(shí)現(xiàn)計(jì)數(shù)器接口防刷功能(升級(jí)版)
- springboot使用redisTemplate操作lua腳本
- Java生態(tài)/Redis中使用Lua腳本的過(guò)程
- Redis中l(wèi)ua腳本實(shí)現(xiàn)及其應(yīng)用場(chǎng)景
- redis使用Lua腳本解決多線程下的超賣(mài)問(wèn)題及原因解析
- redis?lua腳本解決高并發(fā)下秒殺場(chǎng)景
- Redis結(jié)合Lua腳本實(shí)現(xiàn)分布式鎖詳解
- Redis Lua腳本的使用教程
- Redis整合Lua腳本的實(shí)現(xiàn)操作
相關(guān)文章
redis緩存數(shù)據(jù)庫(kù)中數(shù)據(jù)的方法
這篇文章主要為大家詳細(xì)介紹了redis緩存數(shù)據(jù)庫(kù)中數(shù)據(jù)的方法,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-07-07Redis內(nèi)存碎片產(chǎn)生原因及Pipeline管道原理解析
這篇文章主要為大家介紹了Redis內(nèi)存碎片產(chǎn)生原因及Pipeline管道原理解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-03-03Redis中的數(shù)據(jù)結(jié)構(gòu)跳表詳解
跳表是一種基于并聯(lián)的鏈表結(jié)構(gòu),用于在有序元素序列中快速查找元素的數(shù)據(jù)結(jié)構(gòu),本文給大家介紹Redis中的數(shù)據(jù)結(jié)構(gòu)跳表,感興趣的朋友跟隨小編一起看看吧2024-06-06Redis數(shù)據(jù)結(jié)構(gòu)之跳躍表使用學(xué)習(xí)
這篇文章主要為大家介紹了Redis數(shù)據(jù)結(jié)構(gòu)之跳躍表使用學(xué)習(xí),有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-07-07Redis集群指定主從關(guān)系及動(dòng)態(tài)增刪節(jié)點(diǎn)方式
這篇文章主要介紹了Redis集群指定主從關(guān)系及動(dòng)態(tài)增刪節(jié)點(diǎn)方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-01-01