如何重用由lua_newstate创建的lua vm?

时间:2015-06-27 05:44:35

标签: multithreading lua

有时我们需要使用整个lua vm作为工作单位,例如调度 OS线程中的工作单元。但是,它有点开销 lua_newstate和lua_close。之后重用vm是否可行 清理(​​保持沙箱)?

我的简单想法是在vm中创建一个工作线程(为沙盒创建新的_ENV),恢复线程以完成工作,然后删除线程并将vm放入空闲池以供以后重用。

我写了一个简单的程序来验证它。

test.c的:

#include <lua.h>
#include <assert.h>
#include <lauxlib.h>

int main(int argc, char* argv[])
{
  int i = 0;
  const int max = 100000;
  if (argc > 1) {
    for (i = 0; i < max; i++) {
      lua_State* L = luaL_newstate();
      luaL_openlibs(L);
      luaL_loadfile(L, "test.lua");
      lua_resume(L, NULL, 0);
      assert(lua_tointeger(L, 1) == 49995000);
      lua_close(L);
    }
  }
  else {
    lua_State* L = luaL_newstate();
    luaL_openlibs(L);
    luaL_newmetatable(L, "gtm");
    lua_pushglobaltable(L);
    lua_setfield(L, -2, "__index");
    lua_pop(L, 1);
    for (i = 0; i < max; i++) {
      lua_State* co = lua_newthread(L);
      luaL_loadfile(co, "test.lua");
      lua_newtable(co);
      luaL_setmetatable(co, "gtm");
      lua_setupvalue(co, 1, 1);
      lua_resume(co, NULL, 0);
      assert(lua_tointeger(co, 1) == 49995000);
      lua_settop(L, 0);
      lua_gc(L, LUA_GCCOLLECT, 0);
    }
    lua_close(L);
  }
  return 0;
}

test.lua:

local total = 0
for i=0,10000-1 do
  total = total + i
end
return total

输出:

$ gcc -o test test.c -llua -lm -ldl

$ time ./test state

real    0m21.470s
user    0m20.472s
sys 0m0.980s

$ time ./test

real    0m14.933s
user    0m14.036s
sys 0m0.872s

似乎简单的重用只能节省30%的CPU,这远非完美。

有更好的解决方案吗?

0 个答案:

没有答案