在JAVA中使用LUA脚本记,javaj调用lua脚本的函数(转)

最近在做一些奇怪的东西,需要Java应用能够接受用户提交的脚本并执行,网络部分我选择了NanoHTTPD提供基本的HTTP服务器支持,并在Java能承载的许多脚本语言中选择了很久,比如Rhino,Jython和JRuby之类,但它们都太过庞大,并且很难实现沙盒保护服务器环境。最后我的目光投向了Lua,那个被称为粘合剂的语言。遇到的第一个难题是选择所使用的库,纯Java实现的Lua解释器有很多,什么LuaJ,LuaJava,kahlua,还有不知名的mochalua,jill等等(好多好多),其中许多解释器是纯Java实现的,LuaJava则使用了JNI,考虑再三以后我选择了LuaJ,毕竟是纯Java实现,拿来就能用的。
LuaJ也有对应JME和JSE平台的,JSE版是JME版的超集,还带有LuaJava里的luajava模块,能够直接在.lua中调用Java方法,创建Java实例,是很方便的。
折腾了几天,觉得对LuaJ也有足够的了解了,于是把一些相关的代码整理如下:

1
2
3
4
5
6
7
8
9
// 创建一个Lua执行的全局环境。
LuaValue global = JsePlatform.debugGlobals();
 
// 获得loadstring变量,这个变量存储了一个方法,相当于JavaScript里的eval。
LuaValue loadstring = global.get("loadstring");
// 第一个call()方法是调用loadstring这个方法,其参数中使用了LueValue.valueOf()这个静态方法把Java的数据封装成Lua能够使用的数据,第二个call()方法是执行字符串中的表达式,结果是输出了“Hello world!”。
loadstring.call(LuaValue.valueOf("print('Hello world!')")).call();
// 与之类似的还有loadfile,不过它的作用是接受一个文件路径,读入这个文件的内容,执行时调用call。
global.get("loadfile").call("./test.lua").call();

LuaJ直到代码运行结束前都会阻塞线程,这时候开启一个新的线程专门运行即可,但坑爹的是LuaJ运行以后无法中断(即使你中断了它所在的线程),比如你的.lua中有一个while true do end循环,那么你将永远无法中断它,除非退出你的整个Java应用…
怎么样,有没有很坑爹?我谷歌了大半天,发现LuaJ好像是没有官方的解决方案的(同时讨论这类东西的少得可怜!)…我也曾迁移代码到LuaJava上,发现调用了L.close()方法也是不能中断执行,最后终于抓住了一根救命稻草。
这根稻草来自ComputerCraft,一个在MineCraft中模拟计算机的模组,也是使用的LuaJ,但是却能中断一段代码的执行,于是我用jd-gui查看了它的源代码,最终有效实现了LuaJ的执行中中断。

首先容我介绍一下Lua中的一些自带的方法:
debug.sethook()方法能够精确到每一个函数设置钩子回调,这个回调里可以做任何想要做的事情;
coroutine.create()方法能够创建一个协同线程,
coroutine.yield()方法能够暂停这个协同线程(这正是我们想要的),
coroutine.resume()方法用来恢复这个协同线程。
接下来看代码吧:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
package net.airtheva;
 
import java.io.File;
 
import org.luaj.vm2.LuaThread;
import org.luaj.vm2.LuaValue;
import org.luaj.vm2.lib.OneArgFunction;
import org.luaj.vm2.lib.ZeroArgFunction;
import org.luaj.vm2.lib.jse.JsePlatform;
 
public class LuaWorker {
 
  class _Worker implements Runnable {
 
    @Override
    public void run() {
     
      mIsStopping = false;
      mIsStopped = false;
     
      // 产生协同线程。
      mLuaThread = mCoroutineCreate.call(mLoadString.call(LuaValue.valueOf("while true do print('!') end")));
     
      // 执行协同线程(该线程将被阻塞)。
      mCoroutineResume.call(mLuaThread);
     
    }
   
  }
 
  Thread mThread;
 
  LuaValue mGlobal;
 
  LuaValue mLoadString;
 
  LuaValue mDebugSetHook;
 
  LuaValue mNativeCoroutineCreate;
  LuaValue mCoroutineCreate;
  LuaValue mCoroutineYield;
  LuaValue mCoroutineResume;
 
  LuaValue mLuaThread;
 
  boolean mIsStopping = true;
  boolean mIsStopped = true;
 
  public LuaWorker() {
   
    mGlobal = JsePlatform.debugGlobals();
   
    mLoadString = mGlobal.get("loadstring");
   
    mDebugSetHook = mGlobal.get("debug").get("sethook");
   
    LuaValue coroutine = mGlobal.get("coroutine");
    mNativeCoroutineCreate = coroutine.get("create");
    coroutine.set("create", new OneArgFunction() {
     
      @Override
      public LuaValue call(LuaValue value) {
        Debug.L("Called.");
        LuaThread thread = mNativeCoroutineCreate.call(value).checkthread();
        mDebugSetHook.invoke(new LuaValue[] {
            thread,
            new OneArgFunction() {
              @Override
              public LuaValue call(LuaValue value) {
                if(mIsStopping) {
                  //LuaThread.yield(LuaValue.NIL);
                  mCoroutineYield.call(); // 暂停本线程,上面那行也能起到一样的效果。
                  mIsStopped = true;
                }
                return LuaValue.NIL;
              }
            },
            LuaValue.valueOf("crl"), // 这里ComputerCraft用的是LuaValue.NIL,但我这边好像停不下来…
            LuaValue.valueOf(100000) // 这个100000是照着抄的,其实我不知道这是啥意思,等深入使用Lua了应该就会知道了。
        });
        return thread;
      }
     
    });
   
    mCoroutineCreate = coroutine.get("create");
    mCoroutineYield = coroutine.get("yield");
    mCoroutineResume = coroutine.get("resume");
   
  }
 
  public void Start() {
   
    mThread = new Thread(new _Worker());
    mThread.start();
   
  }
 
  public void Stop() {
   
    // 可能回收没做好。
   
    mIsStopping = true;
    mThread.interrupt();
    mThread = null;
   
  }
 
}

然后捣鼓LuaJava的时候发现在eclipse中能够正常运行,导出成.jar以后LuaJava工作不正常,折腾了两天后终于发现原来是编码问题,如果你也出现了问题可以试着指定-Dfile.encoding=UTF-8(坑爹的Windows)。但是因为LuaJava也不知道怎么才能停止,而且它的接口并没有这么丰富,所以最后还是回到了LuaJ的怀抱,这里只是记录一下(毕竟坑了我两天!)。

 

http://airtheva.net/wordpress/?p=159

时间: 2024-10-03 21:48:46

在JAVA中使用LUA脚本记,javaj调用lua脚本的函数(转)的相关文章

java中的类的方法的调用的问题

问题描述 java中的类的方法的调用的问题 新人初学java,有些概念不是很懂,还望各位能帮帮忙,谢谢 在java中,我知道静态方法(变量)可以直接类名.调用,而不用再创建对象, 但是我在学习枚举时遇到这样一个问题: public enum WeekDay{ MON,TUE,WEB,THU,FIR,SAT } public class TestEnum{ public void static main(String [] ,args){ WeekDay today = WeekDay.SAT;

详解Java中用于查找对象哈希码值的hashCode()函数_java

理解hashCode() 的作用是获取哈希码,也称为散列码:它实际上是返回一个int整数.这个哈希码的作用是确定该对象在哈希表中的索引位置. hashCode() 定义在JDK的Object.java中,这就意味着Java中的任何类都包含有hashCode() 函数. 虽然,每个Java类都包含hashCode() 函数.但是,仅仅当创建并某个"类的散列表"(关于"散列表"见下面说明)时,该类的hashCode() 才有用(作用是:确定该类的每一个对象在散列表中的位

VS2005中asp.net用C#怎么调用SAP的rfc函数,请高手指教!

问题描述 用SAPGUI里的COM连接成功了,但是调用时设定连接会出现Badvarianttype的错误,不知是何原因,请高手指教啊!SAPLogonCtrl.SAPLogonControlClasslogon=newSAPLogonCtrl.SAPLogonControlClass();logon.ApplicationServer="10.1.147.2";logon.Client="200";logon.Language="ZH";logo

hadoop hive 操作 是封装在shell脚本好还是java中好?

问题描述 hadoop hive 操作 是封装在shell脚本好还是java中好? 我们现在将hive语句都封装到shell中,因为我个人比较擅长java 所以设想应该可以把hive写在java中,然后再走工作流调用. 就是不知道这两样都有什么优点,希望高手们指点迷津 最好基于你们遇到此问题的做法 多谢 解决方案 高手都在哪 高手都在哪 高手都在哪 解决方案二: 肯定是shell脚本好,shell容易维护,尽量不要使用java,你自己写一堆java代码,过了一段时间后,你发现自己都不认识了,使用

Lua脚本调用外部脚本_Lua

test.lua脚本定义main函数如下: function main(szName, num1, num2) print("main()", szName, num1, num2); local nRandMax = 10000; local nRand = math.random(nRandMax); print("nRand =", nRand) return 1; end 现在我想在test.lua脚本中调用另外一个test1.lua脚本文件中的GetRan

java中的 by value or by reference

java中的基础类型直接存储在栈中,复合类型采用引用类型,把引用也存储在栈中,而对应的对象存储在堆中.因此java中把内存分堆内存和栈内存,在函数中定义的一些基本类型或引用都分配栈内存.堆内存用来存放由new创建的对象和数组,或是static(类装载信息).在堆中分配的内存,由jvm的gc管理.程序只能控制引用的生存期,对象的生存期是jvm控制的. 在java应用程序中,当对象的引用是传递方法一个参数时,传递的时该引用的一个副本,(按值传递).而非引用本身,调用方法的对象引用和副本都是指向同一个

在java中New 类名() 类里面的数据会被重新赋值吗?

问题描述 在java中New 类名() 类里面的数据会被重新赋值吗? 在java中New 类名() 类里面的数据会被重新赋值吗? 解决方案 java中new 类名()是用来调用类的无参构造器开辟内存空间的,初始化一个该类的对象,而该对象里面的值就是类的初始值.会不会被重新赋值要看无参构造器里面的代码有没有再次赋值的过程. 解决方案二: 你这是新建一个 对象 就相当与数据库表 插入一行数据 解决方案三: 是对象里的数据吧,每次new出来的对象里的数据都会有初始化. 解决方案四: 你类里面的都是初始

浅谈Java中的n种随机数产生办法_java

我们从书本上学到什么? 最明显的,也是直观的方式,在Java中生成随机数只要简单的调用: java.lang.Math.random() 在所有其他语言中,生成随机数就像是使用Math工具类,如abs, pow, floor, sqrt和其他数学函数.大多数人通过书籍.教程和课程来了解这个类.一个简单的例子:从0.0到1.0之间可以生成一个双精度浮点数.那么通过上面的信息,开发人员要产生0.0和10.0之间的双精度浮点数会这样来写: Math.random() * 10 而产生0和10之间的整数

浅谈Java中随机数的几种实现方式_java

众所周知,随机数是任何一种编程语言最基本的特征之一.而生成随机数的基本方式也是相同的:产生一个0到1之间的随机数.看似简单,但有时我们也会忽略了一些有趣的功能. 我们从书本上学到什么? 最明显的,也是直观的方式,在Java中生成随机数只要简单的调用: java.lang.Math.random() 在所有其他语言中,生成随机数就像是使用Math工具类,如abs, pow, floor, sqrt和其他数学函数.大多数人通过书籍.教程和课程来了解这个类.一个简单的例子:从0.0到1.0之间可以生成