Java生態(tài)/Redis中使用Lua腳本的過程
一、安裝LUA
Mac上安裝LUA很簡(jiǎn)單,直接使用brew
相關(guān)命令;
brew install lua
使用lua -v
命令可以看到lua已經(jīng)安裝完畢。
1)簡(jiǎn)單使用
創(chuàng)建一個(gè)test.lua文件,內(nèi)容為:
執(zhí)行命令:
lua test.lua
輸出為:
二、lua語(yǔ)法簡(jiǎn)介
Lua 提供了交互式編程和腳本式編程:
- 交互式編程:直接在命令行中輸入語(yǔ)法,可以立即執(zhí)行并查看到執(zhí)行效果。
- 腳本是編程:編寫腳本文件,然后再執(zhí)行。
官方文檔:http://www.lua.org/manual/5.4/
1、注釋
lua提供兩種注釋方式:?jiǎn)涡凶⑨尯投嘈凶⑨?/p>
1)單行注釋
使用兩個(gè)減號(hào);
--
2)多行注釋
--[[ 多行注釋 多行注釋 --]]
2、關(guān)鍵字
下列為 Lua 的保留關(guān)鍵字,和Java一樣 保留關(guān)鍵字不能作為常量或變量。
3、變量
默認(rèn)的情況下,定義一個(gè)變量都是全局變量;如果要用局部變量 需要聲明為local
;
1)全局變量
全局變量不需要聲明,給一個(gè)變量賦值后便創(chuàng)建了這個(gè)全局變量;
訪問一個(gè)沒有初始化的全局變量也不會(huì)出錯(cuò),只不過會(huì)得到結(jié)果:nil
想刪除一個(gè)全局變量,只需要將變量賦值為nil;換言之,當(dāng)且僅當(dāng)一個(gè)變量不等于nil時(shí),這個(gè)變量存在。
此外,一般以下劃線開頭連接一串大寫字母的名字(比如 _VERSION)被保留用于 Lua 內(nèi)部全局變量。
2)局部變量
-- 局部變量賦值 local b=2
4、數(shù)據(jù)類型
Lua 是一個(gè)動(dòng)態(tài)類型語(yǔ)言,變量不要類型定義,只需要為變量賦值。 值可以存儲(chǔ)在變量中,作為參數(shù)傳遞或結(jié)果返回。
Lua 中有 8 個(gè)基本類型分別為:nil、boolean、number、string、userdata、function、thread 和 table。
1)Lua數(shù)組
在Lua 數(shù)組中,索引值是從 1 開始,可以指定為從 0 開始。
2)字符串操作
..
連接兩個(gè)字符串;string.sub()
用于截取字符串;
string.sub(s, i [, j])
- s:要截取的字符串;
- i:截取開始位置;
- j:截取結(jié)束位置,默認(rèn)為 -1,最后一個(gè)字符;
string.find()
用于字符串查找
string.find (str, substr, [init, [plain]])
- 在一個(gè)指定的目標(biāo)字符串 str 中搜索指定的內(nèi)容 substr,如果找到了一個(gè)匹配的子串,就會(huì)返回這個(gè)子串的起始索引和結(jié)束索引,不存在則返回 nil。
init
指定了搜索的起始位置,默認(rèn)為 1,可以一個(gè)負(fù)數(shù),表示從后往前數(shù)的字符個(gè)數(shù)。plain
表示是否使用簡(jiǎn)單模式,默認(rèn)為 false,true 只做簡(jiǎn)單的查找子串的操作,false 表示使用正則模式匹配。
5、if-else
條件表達(dá)式結(jié)果可以是任何值,Lua認(rèn)為false和nil為假,true和非nil為真。
整體的if-else結(jié)構(gòu)和我們使用的高級(jí)語(yǔ)言(Java、GO)類似,區(qū)別在于:LUA中的if()表達(dá)式滿足之后想要做一些其余邏輯,需要緊跟then
,并且流程控制以end
結(jié)尾。
if(xxx) then print("xxx") else if(xx) then print("xx") else print("x") end
6、循環(huán)
1)for循環(huán)
Lua 編程語(yǔ)言中 for語(yǔ)句有兩大類:數(shù)組for循環(huán)、泛型for循環(huán)
1> 數(shù)組for循環(huán)
語(yǔ)法格式:
for var=exp1,exp2,exp3 do <執(zhí)行體> end
var 從 exp1 變化到 exp2,每次變化以 exp3 為步長(zhǎng)遞增 var,并執(zhí)行一次 “執(zhí)行體”。exp3 是可選的,如果不指定,默認(rèn)為1。
2> 泛型for循環(huán)
通過一個(gè)迭代器函數(shù)來遍歷所有值,類似 java 中的 foreach 語(yǔ)句;
語(yǔ)法格式:
--打印數(shù)組a的所有值 a = {"one", "two", "three"} for i, v in ipairs(a) do print(i, v) end
- i 是數(shù)組索引值,v 是對(duì)應(yīng)索引的數(shù)組元素值。
- ipairs是Lua提供的一個(gè)迭代器函數(shù),用來迭代數(shù)組。
2)while循環(huán)
while 循環(huán)語(yǔ)句在判斷條件為 true 時(shí)會(huì)重復(fù)執(zhí)行循環(huán)體語(yǔ)句。
語(yǔ)法格式:
while(condition) do statements end
- statements(循環(huán)體語(yǔ)句) 可以是一條或多條語(yǔ)句,condition(條件) 可以是任意表達(dá)式;
- 在 condition(條件) 為 true 時(shí)執(zhí)行循環(huán)體語(yǔ)句。
3)break提前退出循環(huán)
和Java中的break一個(gè)作用,用于退出當(dāng)前循環(huán)或語(yǔ)句;
7、函數(shù)
在Lua中,函數(shù)是對(duì)語(yǔ)句和表達(dá)式進(jìn)行抽象的主要方法。類似于Java中的方法。
Lua 函數(shù)主要有兩種用途:
- 完成指定的任務(wù),這種情況下函數(shù)作為調(diào)用語(yǔ)句使用;
- 計(jì)算并返回值,這種情況下函數(shù)作為賦值語(yǔ)句的表達(dá)式使用;
函數(shù)的編寫方式如下:
--[[ 函數(shù)返回兩個(gè)值的最大值 --]] 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腳本有兩種方式:字符串的方式、文件的方式;
Java中想要執(zhí)行LUA腳本,首先需要在pom中引入相關(guān)依賴:
<dependency> <groupId>org.luaj</groupId> <artifactId>luaj-jse</artifactId> <version>3.0.1</version> </dependency>
1、字符串方式
對(duì)于簡(jiǎn)單的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(); } }
控制臺(tái)輸出:
2、文件方式
對(duì)于一些比較常用的、復(fù)雜的腳本可以選擇存放在文件中,在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腳本文件:
控制臺(tái)輸出:
3、Luaj概述
Luaj在包裝執(zhí)行具體的Lua代碼時(shí), 有三種不同的模式;
- 純腳本解析執(zhí)行(不選用任何Compiler)
- To Lua字節(jié)碼(LuaC, lua-to-lua-bytecode compiler)(默認(rèn)選用)
- To Java字節(jié)碼(LuaJC, lua-to-java-bytecode compiler)
Luaj中的Globals對(duì)象不是線程安全的, 因此最佳實(shí)踐是每個(gè)線程一個(gè)Globals對(duì)象。
事實(shí)上, 可以采用ThreadLocal的方式來存儲(chǔ)該對(duì)象。
2)性能問題
Lua腳本在JAVA中運(yùn)行,相比于直接運(yùn)行Java代碼會(huì)慢很多,大約1000倍。
四、Redis + Lua(EVAL命令)
在使用Redisson、Jedis+Lua時(shí),我們可以通過redis客戶端集成的、手寫的LUA腳本來保證一系列命令在Redis中可以"原子執(zhí)行"。
在redis執(zhí)行l(wèi)ua腳本時(shí),相當(dāng)于一個(gè)redis級(jí)別的鎖,不能執(zhí)行其他操作,類似于原子操作,這也是redisson實(shí)現(xiàn)的一個(gè)關(guān)鍵點(diǎn)。
比如Redisson中的lua腳本:
Redisson如何實(shí)現(xiàn)分布式鎖,可以看文章:http://www.dbjr.com.cn/article/277312.htm
lua腳本中有如下幾個(gè)概念:
- redis.call():執(zhí)行redis命令。
- KEYS[n]:指腳本中第n個(gè)參數(shù),比如KEYS[1]指腳本中的第一個(gè)參數(shù)。
- ARGV[n]:指腳本中第n個(gè)參數(shù)的值,比如ARGV[1]指腳本中的第一個(gè)參數(shù)的值。
- 返回值中nil與false同一個(gè)意思。
1、EVAL命令
redis2.6.0版本起 采用內(nèi)置的Lua解釋器 通過EVAL命令去執(zhí)行腳本;
redis中的EVAL命令可以用于執(zhí)行一段lua代碼。命令格式如下:
- 第一個(gè)參數(shù)script:表示lua腳本的內(nèi)容;
- 第二參數(shù)numkeys:表示有多少個(gè)鍵值對(duì)。
- 其余參數(shù):先把numkeys個(gè)key列出來,再把numkeys個(gè)arg列出來。
Lua腳本中可以使用2個(gè)函數(shù)調(diào)用redis命令;
- redis.call()
- redis.pcall()
redis.call()與redis.pcall()相似,二者唯一不同之處:
- 如果執(zhí)行的redis命令執(zhí)行失敗,redis.call()將產(chǎn)生一個(gè)Lua error,從而迫使EVAL命令返回一個(gè)錯(cuò)誤給命令的調(diào)用者;
- 然而redis.pcall()將會(huì)捕捉這個(gè)錯(cuò)誤,并返回代表這個(gè)錯(cuò)誤的Lua表。
有那么一段邏輯;
- 如果Redis某個(gè)key的整數(shù)值 和 某個(gè)value相等,則將key對(duì)應(yīng)的整數(shù)值 + 1000;否則將key的值設(shè)置為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é)果如下:
五、總結(jié)
Lua入個(gè)門,知道基本的Lua腳本怎么寫,尤其是Redis + lua的使用。
再到很多網(wǎng)關(guān)中都會(huì)使用一些lua腳本。
到此這篇關(guān)于Java生態(tài)/Redis中如何使用Lua腳本的文章就介紹到這了,更多相關(guān)Redis使用Lua腳本內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
java實(shí)現(xiàn)mongodb的數(shù)據(jù)庫(kù)連接池
這篇文章主要介紹了基于java實(shí)現(xiàn)mongodb的數(shù)據(jù)庫(kù)連接池,Java通過使用mongo-2.7.3.jar包實(shí)現(xiàn)mongodb連接池,感興趣的小伙伴們可以參考一下2015-12-12Java輕松實(shí)現(xiàn)權(quán)限認(rèn)證管理的示例代碼
我們?cè)趯?shí)際開發(fā)中經(jīng)常會(huì)進(jìn)行權(quán)限認(rèn)證管理,給不同的人加上對(duì)應(yīng)的角色和權(quán)限,本文將實(shí)現(xiàn)一個(gè)簡(jiǎn)易的權(quán)限驗(yàn)證管理系統(tǒng),感興趣的小伙伴可以了解下2023-12-12使用idea遠(yuǎn)程調(diào)試jar包的配置過程
這篇文章主要介紹了使用idea遠(yuǎn)程調(diào)試jar包的配置過程,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-09-09java中字符進(jìn)行全角半角轉(zhuǎn)換示例代碼
全角:指一個(gè)字符占用兩個(gè)標(biāo)準(zhǔn)字符位置,而半角:指一字符占用一個(gè)標(biāo)準(zhǔn)的字符位置,在日常開發(fā)中經(jīng)常會(huì)遇到全角半角轉(zhuǎn)換的要求,下面這篇文章主要給大家介紹了關(guān)于java中字符進(jìn)行全角半角轉(zhuǎn)換的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),需要的朋友可以參考下。2017-08-08spring boot項(xiàng)目同時(shí)傳遞參數(shù)和文件的多種方式代碼演示
這篇文章主要介紹了spring boot項(xiàng)目同時(shí)傳遞參數(shù)和文件的多種方式,在開發(fā)接口中,遇到了需要同時(shí)接收參數(shù)和文件的情況,可以有多種方式實(shí)現(xiàn)文件+參數(shù)的接收,這里基于spring boot 3 + vue 3 + axios,做一個(gè)簡(jiǎn)單的代碼演示,需要的朋友可以參考下2023-06-06一文詳解如何配置MyBatis實(shí)現(xiàn)打印可執(zhí)行的SQL語(yǔ)句
在MyBatis中,動(dòng)態(tài)SQL是一個(gè)強(qiáng)大的特性,允許我們?cè)赬ML映射文件或注解中編寫條件語(yǔ)句,根據(jù)運(yùn)行時(shí)的參數(shù)來決定SQL的具體執(zhí)行內(nèi)容,這篇文章主要給大家介紹了關(guān)于如何配置MyBatis實(shí)現(xiàn)打印可執(zhí)行的SQL語(yǔ)句的相關(guān)資料,需要的朋友可以參考下2024-08-08