Java和Lua的完美结合:实现Java程序的动态扩展和脚本自动升级

科技   2025-01-07 11:55   上海  

👉 这是一个或许对你有用的社群

🐱 一对一交流/面试小册/简历优化/求职解惑,欢迎加入芋道快速开发平台知识星球。下面是星球提供的部分资料: 

👉这是一个或许对你有用的开源项目

国产 Star 破 10w+ 的开源项目,前端包括管理后台 + 微信小程序,后端支持单体和微服务架构。

功能涵盖 RBAC 权限、SaaS 多租户、数据权限、商城、支付、工作流、大屏报表、微信公众号等等功能:

  • Boot 仓库:https://gitee.com/zhijiantianya/ruoyi-vue-pro
  • Cloud 仓库:https://gitee.com/zhijiantianya/yudao-cloud
  • 视频教程:https://doc.iocoder.cn
【国内首批】支持 JDK 21 + SpringBoot 3.2.2、JDK 8 + Spring Boot 2.7.18 双版本 

来源:juejin.cn/post/
7273037222436847616


Lua是一种轻量级的脚本语言,常用于游戏开发和嵌入式系统中。它与Java不同,Lua不支持多线程和原生GUI编程。因此,在一些场景下,我们需要将Java和Lua结合起来使用,以弥补两者的不足。

本篇博文将介绍如何在Java程序中使用Lua代码,并且在Lua中调用Java代码。

一、在Java中使用Lua

1.1. 使用LuaJ

LuaJ 是一个 Java 实现的 Lua 解释器,它提供了 Java 与 Lua 之间的桥梁。它的实现原理是使用 JNI 技术将 Java 和 Lua 进行绑定,并提供了 Java 对 Lua 的封装。

具体来说,LuaJ 的实现包括三个部分:

(1)Lua 语言编译器

LuaJ 使用 Lua 语言编写了一组 Lua 编译器,用于将 Lua 代码转换成 Lua 字节码。在 LuaJ 中,编译器并不把 Lua 代码翻译成 Java 代码,而是生成 Lua 字节码。这些字节码可以通过 Java 调用 Lua 解释器来执行。

(2) Lua 虚拟机

LuaJ 为 Java 提供了一个 Lua 虚拟机,可以执行 Lua 字节码。Lua 虚拟机是使用 JNI 接口调用 Lua C 库实现的,它可以运行 Lua 代码和处理 Lua 数据类型。

(3) Java 与 Lua 的桥梁

LuaJ 提供了一组 Java 类库,用于在 Java 中调用 Lua 代码和访问 Lua 数据类型。它提供了 LuaValue 和 LuaFunction 两个关键类,分别对应 Lua 的值和函数。LuaValue 类主要用于表示 Lua 数据类型,包括 Lua 基本类型(nil、boolean、number、string)和 Lua 复杂数据类型(table、function、userdata)。LuaFunction 类则用于表示 Lua 函数,它是一个抽象类,用于封装 Lua 函数的调用。在 Java 中,我们可以使用这些类来调用 Lua 函数和访问 Lua 数据。

(4) Java 与 Lua 的使用

为了在Java中使用Lua,我们需要先引入一个Lua解释器。LuaJ是一个Java实现的Lua解释器,提供了Java与Lua之间的接口。我们可以通过Maven引入LuaJ库:

<dependency>
    <groupId>org.luaj</groupId>
    <artifactId>luaj-jse</artifactId>
    <version>3.0.1</version>
</dependency>

然后,我们就可以开始在Java中使用Lua了。下面是一个简单的例子,展示了如何在Java中执行Lua代码:

import org.luaj.vm2.*;
import org.luaj.vm2.lib.jse.*;

public class HelloWorld {
    public static void main(String[] args) {
        LuaValue globals = JsePlatform.standardGlobals();
        LuaValue chunk = globals.load("print('Hello, World!')");
        chunk.call();
    }
}

在这个例子中,我们首先通过JsePlatform.standardGlobals()方法获取了一个Lua全局环境,然后通过globals.load()方法加载了一段Lua代码,并将其编译成一个Lua函数。最后,我们调用了这个函数,输出了"Hello, World!"。

当需要将 Lua 函数作为参数传递给 Java 方法时,我们可以使用 LuaJ 库提供的 LuaFunction 类来实现。下面我写个简单的用例来展示如何将 Lua 函数作为参数传递给 Java 方法:

import org.luaj.vm2.*;
import org.luaj.vm2.lib.jse.*;

public class LuaJavaExample {
    public static void main(String[] args) {
        LuaValue globals = JsePlatform.standardGlobals();

        // 定义 Lua 函数
        LuaValue luaFunction = LuaValue.valueOf(new TwoParametersFunction());

        // 调用 Java 方法,并传递 Lua 函数作为参数
        invokeJavaMethod(luaFunction);

        // 在 Lua 中调用 Java 方法
        globals.set("invokeJavaMethod"new InvokeJavaMethodFunction());
        globals.load("invokeJavaMethod()").call();
    }

    public static void invokeJavaMethod(LuaValue luaFunction) {
        // 在 Java 方法中调用传递进来的 Lua 函数
        luaFunction.call(LuaValue.valueOf("Hello"), LuaValue.valueOf("World"));
    }

    public static class TwoParametersFunction extends OneArgFunction {
        @Override
        public LuaValue call(LuaValue arg) {
            String firstParameter = arg.checkjstring();
            String secondParameter = arg.checkjstring(2);
            System.out.println("First parameter: " + firstParameter);
            System.out.println("Second parameter: " + secondParameter);
            return LuaValue.NIL;
        }
    }

    public static class InvokeJavaMethodFunction extends ZeroArgFunction {
        @Override
        public LuaValue call() {
            // 在 Lua 中调用 Java 方法
            invokeJavaMethod(LuaValue.valueOf(new TwoParametersFunction()));
            return LuaValue.NIL;
        }
    }
}

在这个示例中,我定义了一个 Lua 函数TwoParametersFunction,它继承自OneArgFunction,用于接收两个参数。在 Java 的InvokeJavaMethodFunction类中,我使用LuaValue.valueOf(new TwoParametersFunction())将 Lua 函数转换为 LuaValue 对象,并通过invokeJavaMethod()方法传递给 Java 方法。

其中,invokeJavaMethod()方法在 Java 中调用传递进来的 Lua 函数,示例中传递了两个参数。TwoParametersFunction类负责处理传入的参数并进行相应的操作,这里只是简单地打印出两个参数值。

最后,我在 Lua 环境中定义了一个全局函数invokeJavaMethod(),用于在 Lua 中调用 Java 方法。在 Lua 中,我载入了此 Lua 脚本并调用invokeJavaMethod()函数,它会再次调用 Java 的invokeJavaMethod()方法,从而形成一个循环调用的结构。

1.2. 实现动态扩展和脚本自动升级

(1)实现动态扩展

动态扩展是指在不停止或重新编译Java程序的情况下,通过加载并执行Lua脚本来增加程序的功能或修改程序的行为。

下面是一个示例,演示了如何使用LuaJ实现动态扩展功能:

import org.luaj.vm2.*;
import org.luaj.vm2.lib.jse.*;

public class DynamicExtensionExample {
    public static void main(String[] args) {
        LuaValue globals = JsePlatform.standardGlobals();

        // 加载并执行Lua脚本
        globals.loadfile("extension.lua").call();

        // 调用Lua函数
        LuaValue luaFunction = globals.get("addTwoNumbers");
        LuaValue result = luaFunction.call(1020);

        // 打印结果
        System.out.println("Result: " + result.toint());
    }
}

在上面的示例中,我们首先创建了一个Lua环境,并加载了标准的全局函数和库。然后,我们使用globals.loadfile("extension.lua").call()加载并执行了一个名为extension.lua的Lua脚本。

在Lua脚本中,我们可以定义新的函数或修改现有函数,以实现对Java程序的扩展。在本例中,我们假设extension.lua文件中定义了一个名为addTwoNumbers的Lua函数,该函数接收两个参数并返回它们的和。

在Java程序中,我们可以通过globals.get("addTwoNumbers")获取到这个Lua函数,并使用luaFunction.call(10, 20)调用它。最后,我们打印出函数的返回值。

通过这样的方式,我们可以将一些核心的功能逻辑写成Lua脚本,通过加载和执行脚本来实现Java程序的动态扩展。这使得程序的修改和功能的增加变得非常灵活和方便。

(2)实现脚本自动升级

脚本自动升级是指在Java程序运行过程中,根据特定条件自动检测并加载新版本的Lua脚本,以实现程序的自动更新。

下面是一个示例,演示了如何使用LuaJ实现脚本自动升级功能:

import org.luaj.vm2.*;
import org.luaj.vm2.lib.jse.*;

public class ScriptAutoUpgradeExample {
    public static void main(String[] args) {
        LuaValue globals = JsePlatform.standardGlobals();

        // 加载并执行初始版本的Lua脚本
        globals.loadfile("script.lua").call();

        // 循环检测是否有新版本的Lua脚本
        while (true) {
            // 检测是否有新版本的脚本

            // 如果有新版本,则加载并执行新版本的Lua脚本
            if (hasNewScriptVersion()) {
                globals.loadfile("new_script.lua").call();
            }

            // 执行其他程序逻辑

            // 休眠一段时间后再次进行检测
            sleep(1000);
        }
    }

    // 检测是否有新版本的脚本
    private static boolean hasNewScriptVersion() {
        // 实现自己的检测逻辑,如从远程服务器下载新版本脚本进行比对等
        // 返回 true 表示有新版本的脚本可用,否则返回 false
        return false;
    }

    // 线程休眠
    private static void sleep(long milliseconds) {
        try {
            Thread.sleep(milliseconds);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
    }
}

在上面的示例中,我们首先创建了一个Lua环境,并加载了标准的全局函数和库。然后,我们使用globals.loadfile("script.lua").call()加载并执行了初始版本的Lua脚本。

接下来,我们进入一个无限循环,不断检测是否有新版本的脚本可用。在hasNewScriptVersion()方法中,你可以根据自己的需求实现检测逻辑。如果有新版本的脚本可用,我们使用globals.loadfile("new_script.lua").call()加载并执行新版本的Lua脚本。

通过这样的方式,我们可以在程序运行期间检测并自动加载新版本的Lua脚本,实现Java程序的脚本自动升级功能。

基于 Spring Boot + MyBatis Plus + Vue & Element 实现的后台管理系统 + 用户小程序,支持 RBAC 动态权限、多租户、数据权限、工作流、三方登录、支付、短信、商城等功能

  • 项目地址:https://github.com/YunaiV/ruoyi-vue-pro
  • 视频教程:https://doc.iocoder.cn/video/

二、在Lua中使用Java

2.1. 使用Lua Java库

与在Java中使用Lua类似,我们也需要引入一个Java与Lua之间的接口库。Lua Java是一个Java实现的Lua接口库,它允许我们在Lua脚本中访问Java对象和方法。我们可以通过Maven引入LuaJava库:

<dependency>
    <groupId>com.naef.jnlua</groupId>
    <artifactId>jnlua</artifactId>
    <version>0.9.0</version>
</dependency>

然后,在Lua脚本中就可以使用Java对象和方法了。下面是一个例子,展示了如何在Lua中访问Java对象:

import org.luaj.vm2.*;
import org.luaj.vm2.lib.*;
import com.naef.jnlua.*;

public class HelloWorld {
    public static void main(String[] args) throws Exception {
        LuaState luaState = JNLuaUtil.newState();
        luaState.openLibs();
        luaState.pushJavaObject(new Hello());
        luaState.setGlobal("hello");
        luaState.load("hello:sayHello('World')");
        luaState.call(00);
    }

    public static class Hello {
        public void sayHello(String name) {
            System.out.println("Hello, " + name);
        }
    }
}

在这个例子中,我们创建了一个Java对象Hello,并且将其压入Lua栈中。然后,我们将这个Java对象绑定到名为"hello"的全局变量中:

luaState.pushJavaObject(new Hello());
luaState.setGlobal("hello");

最后,我们在Lua脚本中调用了这个Java对象的方法:

luaState.load("hello:sayHello('World')");
luaState.call(00);

在这个例子中,我们使用了Lua的冒号语法来调用Java方法。

2.2. 在Lua中访问Java类

除了访问Java对象,我们还可以在Lua中访问Java类。下面是一个例子,展示了如何在Lua中访问Java类,并调用其静态方法:

import org.luaj.vm2.*;
import org.luaj.vm2.lib.*;
import com.naef.jnlua.*;

public class HelloWorld {
    public static void main(String[] args) throws Exception {
        LuaState luaState = JNLuaUtil.newState();
        luaState.openLibs();
        luaState.pushJavaClass(Hello.class);
        luaState.setGlobal("Hello");
        luaState.load("Hello.sayHello('World')");
        luaState.call(00);
    }

    public static class Hello {
        public static void sayHello(String name) {
            System.out.println("Hello, " + name);
        }
    }
}

在这个例子中,我们使用了Lua的pushJavaClass()方法将Java类Hello压入Lua栈中,并且将其绑定到名为"Hello"的全局变量中:

luaState.pushJavaClass(Hello.class);
luaState.setGlobal("Hello");

最后,我们在Lua脚本中调用了Hello类的静态方法:

luaState.load("Hello.sayHello('World')");
luaState.call(00);

与访问Java对象一样,我们可以使用Lua语法来调用Java类和方法。

基于 Spring Cloud Alibaba + Gateway + Nacos + RocketMQ + Vue & Element 实现的后台管理系统 + 用户小程序,支持 RBAC 动态权限、多租户、数据权限、工作流、三方登录、支付、短信、商城等功能

  • 项目地址:https://github.com/YunaiV/yudao-cloud
  • 视频教程:https://doc.iocoder.cn/video/

三、小结一下

本篇博文介绍了如何在Java中使用Lua和在Lua中使用Java。这两种方案都需要引入一个Java与Lua之间的接口库,分别是LuaJ和LuaJava。在Java中使用Lua,我们需要通过LuaJ库来执行Lua代码,并且在Lua全局环境中添加Java方法。

在Lua中使用Java,我们需要通过LuaJava库来访问Java对象和方法。这两种方案都可以帮助我们弥补Java和Lua各自的不足,提高程序的灵活性和可扩展性。


欢迎加入我的知识星球,全面提升技术能力。

👉 加入方式,长按”或“扫描”下方二维码噢

星球的内容包括:项目实战、面试招聘、源码解析、学习路线。

文章有帮助的话,在看,转发吧。

谢谢支持哟 (*^__^*)

Java基基
一个苦练基本功的 Java 公众号,所以取名 Java 基基
 最新文章