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