Lua与Java相互交互的例子

来源:互联网 发布:犀牛软件 垃圾袋 编辑:程序博客网 时间:2024/06/06 01:28
该例子用到一个Java文件,一个Lua脚本文件,引用了Apache commons.lang 第三方jar包,以及luajava.jar,相关jar包请自行下载。

LuaDemo.java

package com.chilijoy.lua;

import java.io.File;
import java.util.Queue;
import java.util.Random;
import java.util.concurrent.ConcurrentLinkedQueue;

import org.apache.commons.lang3.RandomStringUtils;
import org.keplerproject.luajava.LuaState;
import org.keplerproject.luajava.LuaStateFactory;

/*******************************************************************************
*
* LuaDemo.java Created on 2014年7月28日
*
* Author: linfenliang
*
* Description:
*
* Version: 1.0
******************************************************************************/

public class LuaDemo {
public static final String luaPath = "." + File.separator + "conf" + File.separator ;
public static Queue<String> queue = new ConcurrentLinkedQueue<String>();
public static void main(String[] args) {
LuaState state = LuaStateFactory.newLuaState();
state.openLibs();
state.LdoFile(luaPath+"demo.lua");

new PublishMessage(state).start();
new SubscribeMessage(queue).start();
}


}
class PublishMessage extends Thread{
private LuaState state;
public PublishMessage(LuaState state) {
this.state = state;
}
@Override
public void run() {
while(true){
String value = RandomStringUtils.randomAlphabetic(32);
state.getField(LuaState.LUA_GLOBALSINDEX, "pushDataToQueue");
state.pushString(value);
state.call(1, 0);
try {
int time = (int)(new Random().nextDouble()*10);
time = time>0?time:3;
System.out.println("写入数据:"+value+",暂停 "+ time +" 秒");
Thread.sleep(time*1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
}
class SubscribeMessage extends Thread{
private Queue<String> queue;
public SubscribeMessage(Queue<String> queue) {
this.queue = queue;
}
@Override
public void run() {
int time = 3;
while(true){
try {
Thread.sleep(time*1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(time+"秒取一次数据:"+this.queue.poll());
}
}
}

demo.lua

LuaDemo = luajava.bindClass("com.chilijoy.lua.LuaDemo");
function pushDataToQueue(data)
LuaDemo.queue:offer(data);
end

打印结果(示例)

写入数据:iQRYgmsRCKVggBNlCouEsTNLHmhKlHlT,暂停 8 秒
3秒取一次数据:iQRYgmsRCKVggBNlCouEsTNLHmhKlHlT
3秒取一次数据:null
写入数据:YSffHtLxknhgDoIRGzzSoBoqlzAjResi,暂停 6 秒
3秒取一次数据:YSffHtLxknhgDoIRGzzSoBoqlzAjResi
3秒取一次数据:null
写入数据:ubikJSKwedEhKalsigLvTCVEJEnLRiXN,暂停 3 秒
3秒取一次数据:ubikJSKwedEhKalsigLvTCVEJEnLRiXN
写入数据:YRrazjaOHVWZIIbMrBbUgVLrfSsVWDQm,暂停 4 秒
3秒取一次数据:YRrazjaOHVWZIIbMrBbUgVLrfSsVWDQm
3秒取一次数据:null
写入数据:zhuLoDjSJMpInVPwmCYkshRtWpLdIPwS,暂停 1 秒
写入数据:fnKpyQlsznsEmZHjtDvZDPOpzVHyNTtJ,暂停 8 秒
3秒取一次数据:zhuLoDjSJMpInVPwmCYkshRtWpLdIPwS


总结:

Lua与Java的交互:

Java调用Lua:
采用luajava.jar包 API调用实现参数传递(注意:变长参数不可)
Lua调用Java:
(1) luajava.bindClass("package.ClassName");
引用的方式调用Java中的对象(注意:不可调用Java的中数组形式如 int[]等);
(2)luajava.newInstance("package.ClassName",parameter);
实例化调用Java中的对象,后面可接多个参数传入Java实例化方法;

Lua将数据传给Java,以便Java可以返回给客户端,此时,应将Lua处理完的业务数据通过引用Java对象的方式写入Java的队列中,然后Java可定时读取该队列完成数据的通信(同样是走RPC调用),
该种方式调用存在一定的延迟,好处是在Lua处理业务数据过程中,可以将不同的数据写入不同的队列中,而无需等待所有业务都完成处理再返回给Java处理;另外一种方式为待所有的业务逻辑在Lua中完成处理,
通过luajava.jar包的提供的API接收返回结果,该种方式可以在Lua完成数据处理后立即接收到返回的结果,进行下一步数据处理,然而如果Lua处理数据耗时较长,则此处会出现长时间阻塞等待情况,自行斟酌选择。
0 0