您好,登錄后才能下訂單哦!
本篇內(nèi)容主要講解“Java生態(tài)/Redis中怎么使用Lua腳本”,感興趣的朋友不妨來看看。本文介紹的方法操作簡單快捷,實用性強。下面就讓小編來帶大家學習“Java生態(tài)/Redis中怎么使用Lua腳本”吧!
Mac上安裝LUA很簡單,直接使用brew
相關(guān)命令;
brew install lua
使用lua -v
命令可以看到lua已經(jīng)安裝完畢。
1)簡單使用
創(chuàng)建一個test.lua文件,內(nèi)容為:
執(zhí)行命令:
lua test.lua
輸出為:
Lua 提供了交互式編程和腳本式編程:
交互式編程:直接在命令行中輸入語法,可以立即執(zhí)行并查看到執(zhí)行效果。
腳本是編程:編寫腳本文件,然后再執(zhí)行。
lua提供兩種注釋方式:單行注釋和多行注釋
使用兩個減號;
--
--[[ 多行注釋 多行注釋 --]]
下列為 Lua 的保留關(guān)鍵字,和Java一樣 保留關(guān)鍵字不能作為常量或變量。
默認的情況下,定義一個變量都是全局變量;如果要用局部變量 需要聲明為local
;
全局變量不需要聲明,給一個變量賦值后便創(chuàng)建了這個全局變量;
訪問一個沒有初始化的全局變量也不會出錯,只不過會得到結(jié)果:nil
想刪除一個全局變量,只需要將變量賦值為nil;換言之,當且僅當一個變量不等于nil時,這個變量存在。
此外,一般以下劃線開頭連接一串大寫字母的名字(比如 _VERSION)被保留用于 Lua 內(nèi)部全局變量。
-- 局部變量賦值 local b=2
Lua 是一個動態(tài)類型語言,變量不要類型定義,只需要為變量賦值。 值可以存儲在變量中,作為參數(shù)傳遞或結(jié)果返回。
Lua 中有 8 個基本類型分別為:nil、boolean、number、string、userdata、function、thread 和 table。
在Lua 數(shù)組中,索引值是從 1 開始,可以指定為從 0 開始。
..
連接兩個字符串;
string.sub()
用于截取字符串;
string.sub(s, i [, j])
s:要截取的字符串;
i:截取開始位置;
j:截取結(jié)束位置,默認為 -1,最后一個字符;
string.find()
用于字符串查找
string.find (str, substr, [init, [plain]])
在一個指定的目標字符串 str 中搜索指定的內(nèi)容 substr,如果找到了一個匹配的子串,就會返回這個子串的起始索引和結(jié)束索引,不存在則返回 nil。
init
指定了搜索的起始位置,默認為 1,可以一個負數(shù),表示從后往前數(shù)的字符個數(shù)。
plain
表示是否使用簡單模式,默認為 false,true 只做簡單的查找子串的操作,false 表示使用正則模式匹配。
條件表達式結(jié)果可以是任何值,Lua認為false和nil為假,true和非nil為真。
整體的if-else結(jié)構(gòu)和我們使用的高級語言(Java、GO)類似,區(qū)別在于:LUA中的if()表達式滿足之后想要做一些其余邏輯,需要緊跟then
,并且流程控制以end
結(jié)尾。
if(xxx) then print("xxx") else if(xx) then print("xx") else print("x") end
Lua 編程語言中 for語句有兩大類:數(shù)組for循環(huán)、泛型for循環(huán)
1> 數(shù)組for循環(huán)
語法格式:
for var=exp1,exp2,exp3 do <執(zhí)行體> end
var 從 exp1 變化到 exp2,每次變化以 exp3 為步長遞增 var,并執(zhí)行一次 “執(zhí)行體”。exp3 是可選的,如果不指定,默認為1。
2> 泛型for循環(huán)
通過一個迭代器函數(shù)來遍歷所有值,類似 java 中的 foreach 語句;
語法格式:
--打印數(shù)組a的所有值 a = {"one", "two", "three"} for i, v in ipairs(a) do print(i, v) end
i 是數(shù)組索引值,v 是對應索引的數(shù)組元素值。
ipairs是Lua提供的一個迭代器函數(shù),用來迭代數(shù)組。
while 循環(huán)語句在判斷條件為 true 時會重復執(zhí)行循環(huán)體語句。
語法格式:
while(condition) do statements end
statements(循環(huán)體語句) 可以是一條或多條語句,condition(條件) 可以是任意表達式;
在 condition(條件) 為 true 時執(zhí)行循環(huán)體語句。
和Java中的break一個作用,用于退出當前循環(huán)或語句;
在Lua中,函數(shù)是對語句和表達式進行抽象的主要方法。類似于Java中的方法。
Lua 函數(shù)主要有兩種用途:
完成指定的任務,這種情況下函數(shù)作為調(diào)用語句使用;
計算并返回值,這種情況下函數(shù)作為賦值語句的表達式使用;
函數(shù)的編寫方式如下:
--[[ 函數(shù)返回兩個值的最大值 --]] function max(num1, num2) if (num1 > num2) then result = num1; else result = num2; end return result; end -- 調(diào)用函數(shù) print("兩值比較最大值為 ",max(10,4)) print("兩值比較最大值為 ",max(5,6))
Java中執(zhí)行Lua腳本有兩種方式:字符串的方式、文件的方式;
Java中想要執(zhí)行LUA腳本,首先需要在pom中引入相關(guān)依賴:
<dependency> <groupId>org.luaj</groupId> <artifactId>luaj-jse</artifactId> <version>3.0.1</version> </dependency>
對于簡單的lua腳本,可以直接用java字符串寫;
package com.saint.base.lua; import org.luaj.vm2.Globals; import org.luaj.vm2.LuaValue; import org.luaj.vm2.lib.jse.JsePlatform; public class LuaString { public static void main(String[] args) { String luaStr = "print 'Saint is best man'"; Globals globals = JsePlatform.standardGlobals(); LuaValue luaValue = globals.load(luaStr); luaValue.call(); } }
控制臺輸出:
對于一些比較常用的、復雜的腳本可以選擇存放在文件中,在Java中再調(diào)用lua文件;
package com.saint.base.lua; import org.luaj.vm2.Globals; import org.luaj.vm2.LuaValue; import org.luaj.vm2.lib.jse.JsePlatform; import java.io.FileNotFoundException; public class LuaFile { public static void main(String[] args) throws FileNotFoundException { // lua腳本的文件路徑 String luaPath = "/xxxx/javaTest.lua"; Globals globals = JsePlatform.standardGlobals(); //加載腳本文件login.lua,并編譯 globals.loadfile(luaPath).call(); LuaValue func1 = globals.get(LuaValue.valueOf("print1")); func1.call(); LuaValue func2 = globals.get(LuaValue.valueOf("print2")); String luaResp = func2.call(LuaValue.valueOf("saint-input-param")).toString(); System.out.println("lua file return is : " + luaResp); } }
lua腳本文件:
控制臺輸出:
Luaj在包裝執(zhí)行具體的Lua代碼時, 有三種不同的模式;
純腳本解析執(zhí)行(不選用任何Compiler)
To Lua字節(jié)碼(LuaC, lua-to-lua-bytecode compiler)(默認選用)
To Java字節(jié)碼(LuaJC, lua-to-java-bytecode compiler)
Luaj中的Globals對象不是線程安全的, 因此最佳實踐是每個線程一個Globals對象。
事實上, 可以采用ThreadLocal的方式來存儲該對象。
2)性能問題
Lua腳本在JAVA中運行,相比于直接運行Java代碼會慢很多,大約1000倍。
在使用Redisson、Jedis+Lua時,我們可以通過redis客戶端集成的、手寫的LUA腳本來保證一系列命令在Redis中可以"原子執(zhí)行"。
在redis執(zhí)行l(wèi)ua腳本時,相當于一個redis級別的鎖,不能執(zhí)行其他操作,類似于原子操作,這也是redisson實現(xiàn)的一個關(guān)鍵點。
比如Redisson中的lua腳本:
Redisson如何實現(xiàn)分布式鎖,可以看文章:http://www.kemok4.com/article/277312.htm
lua腳本中有如下幾個概念:
redis.call():執(zhí)行redis命令。
KEYS[n]:指腳本中第n個參數(shù),比如KEYS[1]指腳本中的第一個參數(shù)。
ARGV[n]:指腳本中第n個參數(shù)的值,比如ARGV[1]指腳本中的第一個參數(shù)的值。
返回值中nil與false同一個意思。
redis2.6.0版本起 采用內(nèi)置的Lua解釋器 通過EVAL命令去執(zhí)行腳本;
redis中的EVAL命令可以用于執(zhí)行一段lua代碼。命令格式如下:
第一個參數(shù)script:表示lua腳本的內(nèi)容;
第二參數(shù)numkeys:表示有多少個鍵值對。
其余參數(shù):先把numkeys個key列出來,再把numkeys個arg列出來。
Lua腳本中可以使用2個函數(shù)調(diào)用redis命令;
redis.call()
redis.pcall()
redis.call()與redis.pcall()相似,二者唯一不同之處:
如果執(zhí)行的redis命令執(zhí)行失敗,redis.call()將產(chǎn)生一個Lua error,從而迫使EVAL命令返回一個錯誤給命令的調(diào)用者;
然而redis.pcall()將會捕捉這個錯誤,并返回代表這個錯誤的Lua表。
有那么一段邏輯;
如果Redis某個key的整數(shù)值 和 某個value相等,則將key對應的整數(shù)值 + 1000;否則將key的值設置為9999;
lua腳本執(zhí)行命令如下:
EVAL "if redis.call('get', KEYS[1]) == ARGV[1] then return redis.call('INCRBY', KEYS[1], 1000); else redis.call('set', KEYS[1], 9999); return nil; end;" 1 test 100
根據(jù)test值的不同,不同的執(zhí)行結(jié)果如下:
到此,相信大家對“Java生態(tài)/Redis中怎么使用Lua腳本”有了更深的了解,不妨來實際操作一番吧!這里是億速云網(wǎng)站,更多相關(guān)內(nèi)容可以進入相關(guān)頻道進行查詢,關(guān)注我們,繼續(xù)學習!
免責聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。