Javaエコシステム/RedisでLuaスクリプトを使用する方法
1. LUA のインストール
Mac への LUA のインストールは非常に簡単で、brew
関連コマンドを直接使用するだけです。
コマンドを使用すると、lua がインストールされていることを確認できます。
を使用して、次の内容の test.lua ファイルを作成します。
brew install lua
出力は次のとおりです:
2. lua 構文の概要
Lua は対話型プログラミングを提供します式プログラミング:
- #スクリプトはプログラミングです。スクリプト ファイルを作成して実行します。
- #1. コメント
- lua は、単一行コメントと複数行コメントという 2 つのコメント メソッドを提供します1) 単一行コメント
--
2) 複数行のコメントlua test.lua
は以下の通り Lua の予約キーワードは Java と同様に定数や変数として使用できません。
3. 変数
デフォルトでは、定義された変数はグローバル変数です; ローカル変数を使用したい場合は、それを次のように宣言する必要があります。
local;
初期化されていないグローバル変数へのアクセス エラーは発生しませんが、結果は次のようになります: nil
変数に値 nil が割り当てられている限り、グローバル変数は削除できます; つまり、変数が nil の場合にのみ、変数は存在しません。
2) ローカル変数
--[[ 多行注释 多行注释 --]]
Lua は動的に型指定される言語です。変数は型指定する必要はなく、変数だけを型指定する必要があります。割り当てられた値。値は変数に格納したり、引数として渡したり、結果として返すことができます。
Lua には、nil、boolean、number、string、userdata、function、thread、table の 8 つの基本型があります。1) Lua 配列
Lua 配列では、インデックス値は 1 から始まり、0 から始まるように指定できます。
.. 2 つの文字列を接続します;
- string.sub()
は文字列をインターセプトするために使用されます;
-- 局部变量赋值 local b=2
ログイン後にコピー s: インターセプトされる文字 String;
i: インターセプトの開始位置;
j: インターセプトの終了位置、デフォルトは -1、最後の文字;
- string.find() 文字列検索に使用されます
string.sub(s, i [, j])
ログイン後にコピー指定されたターゲット文字列内の指定された文字列を検索します str 内容 substr. If一致する部分文字列が見つかった場合は、その部分文字列の開始インデックスと終了インデックスが返され、存在しない場合は nil が返されます。
- 検索の開始位置を指定します。デフォルトは 1 です。後ろから前までの文字数を示す負の数も可能です。
- #plain
シンプル モードを使用するかどうかを示します。デフォルトは false、true は部分文字列の単純な検索のみを行い、false は通常のパターン マッチングを使用することを示します。
5. if-else 条件式の結果は任意の値になります。Lua は、false と nil を false と見なし、true と非 nil を false とみなします。本当だ。
then全体的な if-else 構造は、私たちが使用する高級言語 (Java、GO) に似ています。違いは、LUA の if() 式が満たされた後に他のロジックを実行したい場合であることです。
に従う必要があり、フロー制御は
で終了します。
string.find (str, substr, [init, [plain]])
6. ループ
1) for ループLua プログラミング言語の for ステートメントには 2 つの主要なカテゴリがあります: 配列 for ループと汎用 for ループ
1> ループ用の配列
if(xxx) then print("xxx") else if(xx) then print("xx") else print("x") 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提供的一个迭代器函数,用来迭代数组。
2)while循环
while 循环语句在判断条件为 true 时会重复执行循环体语句。
语法格式:
while(condition) do statements end
statements(循环体语句) 可以是一条或多条语句,condition(条件) 可以是任意表达式;
在 condition(条件) 为 true 时执行循环体语句。
3)break提前退出循环
和Java中的break一个作用,用于退出当前循环或语句;
7、函数
在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脚本有两种方式:字符串的方式、文件的方式;
Java中想要执行LUA脚本,首先需要在pom中引入相关依赖:
<dependency> <groupId>org.luaj</groupId> <artifactId>luaj-jse</artifactId> <version>3.0.1</version> </dependency>
1、字符串方式
对于简单的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(); } }
控制台输出:
2、文件方式
对于一些比较常用的、复杂的脚本可以选择存放在文件中,在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脚本文件:
控制台输出:
3、Luaj概述
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倍。
四、Redis + Lua(EVAL命令)
在使用Redisson、Jedis+Lua时,我们可以通过redis客户端集成的、手写的LUA脚本来保证一系列命令在Redis中可以"原子执行"。
在redis执行lua脚本时,相当于一个redis级别的锁,不能执行其他操作,类似于原子操作,这也是redisson实现的一个关键点。
比如Redisson中的lua脚本:
Redisson如何实现分布式锁,可以看文章:https://www.yisu.com/article/277312.htm
lua脚本中有如下几个概念:
redis.call():执行redis命令。
KEYS[n]:指脚本中第n个参数,比如KEYS[1]指脚本中的第一个参数。
ARGV[n]:指脚本中第n个参数的值,比如ARGV[1]指脚本中的第一个参数的值。
返回值中nil与false同一个意思。
1、EVAL命令
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スクリプトを使用する方法の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

ホットAIツール

Undresser.AI Undress
リアルなヌード写真を作成する AI 搭載アプリ

AI Clothes Remover
写真から衣服を削除するオンライン AI ツール。

Undress AI Tool
脱衣画像を無料で

Clothoff.io
AI衣類リムーバー

AI Hentai Generator
AIヘンタイを無料で生成します。

人気の記事

ホットツール

メモ帳++7.3.1
使いやすく無料のコードエディター

SublimeText3 中国語版
中国語版、とても使いやすい

ゼンドスタジオ 13.0.1
強力な PHP 統合開発環境

ドリームウィーバー CS6
ビジュアル Web 開発ツール

SublimeText3 Mac版
神レベルのコード編集ソフト(SublimeText3)

ホットトピック









PHP and Python each have their own advantages, and the choice should be based on project requirements. 1.PHPは、シンプルな構文と高い実行効率を備えたWeb開発に適しています。 2。Pythonは、簡潔な構文とリッチライブラリを備えたデータサイエンスと機械学習に適しています。

Redisクラスターモードは、シャードを介してRedisインスタンスを複数のサーバーに展開し、スケーラビリティと可用性を向上させます。構造の手順は次のとおりです。異なるポートで奇妙なRedisインスタンスを作成します。 3つのセンチネルインスタンスを作成し、Redisインスタンスを監視し、フェールオーバーを監視します。 Sentinel構成ファイルを構成し、Redisインスタンス情報とフェールオーバー設定の監視を追加します。 Redisインスタンス構成ファイルを構成し、クラスターモードを有効にし、クラスター情報ファイルパスを指定します。各Redisインスタンスの情報を含むnodes.confファイルを作成します。クラスターを起動し、CREATEコマンドを実行してクラスターを作成し、レプリカの数を指定します。クラスターにログインしてクラスター情報コマンドを実行して、クラスターステータスを確認します。作る

Redis指令を使用するには、次の手順が必要です。Redisクライアントを開きます。コマンド(動詞キー値)を入力します。必要なパラメーターを提供します(指示ごとに異なります)。 Enterを押してコマンドを実行します。 Redisは、操作の結果を示す応答を返します(通常はOKまたは-ERR)。

Redisデータをクリアする方法:Flushallコマンドを使用して、すべての重要な値をクリアします。 FlushDBコマンドを使用して、現在選択されているデータベースのキー値をクリアします。 [選択]を使用してデータベースを切り替え、FlushDBを使用して複数のデータベースをクリアします。 DELコマンドを使用して、特定のキーを削除します。 Redis-CLIツールを使用してデータをクリアします。

Redisソースコードを理解する最良の方法は、段階的に進むことです。Redisの基本に精通してください。開始点として特定のモジュールまたは機能を選択します。モジュールまたは機能のエントリポイントから始めて、行ごとにコードを表示します。関数コールチェーンを介してコードを表示します。 Redisが使用する基礎となるデータ構造に精通してください。 Redisが使用するアルゴリズムを特定します。

Redisのキューを読むには、キュー名を取得し、LPOPコマンドを使用して要素を読み、空のキューを処理する必要があります。特定の手順は次のとおりです。キュー名を取得します:「キュー:キュー」などの「キュー:」のプレフィックスで名前を付けます。 LPOPコマンドを使用します。キューのヘッドから要素を排出し、LPOP Queue:My-Queueなどの値を返します。空のキューの処理:キューが空の場合、LPOPはnilを返し、要素を読む前にキューが存在するかどうかを確認できます。

Redisはハッシュテーブルを使用してデータを保存し、文字列、リスト、ハッシュテーブル、コレクション、注文コレクションなどのデータ構造をサポートします。 Redisは、スナップショット(RDB)を介してデータを維持し、書き込み専用(AOF)メカニズムを追加します。 Redisは、マスタースレーブレプリケーションを使用して、データの可用性を向上させます。 Redisは、シングルスレッドイベントループを使用して接続とコマンドを処理して、データの原子性と一貫性を確保します。 Redisは、キーの有効期限を設定し、怠zyな削除メカニズムを使用して有効期限キーを削除します。

Redisサーバーを起動する手順には、以下が含まれます。オペレーティングシステムに従ってRedisをインストールします。 Redis-Server(Linux/Macos)またはRedis-Server.exe(Windows)を介してRedisサービスを開始します。 Redis-Cli ping(Linux/macos)またはRedis-Cli.exePing(Windows)コマンドを使用して、サービスステータスを確認します。 Redis-Cli、Python、node.jsなどのRedisクライアントを使用して、サーバーにアクセスします。
