调用 newthread 后如何获取线程索引?

在 JNLUA 中,newThread() 是 Java 中的一个无返回值函数,但我并不太理解实现该函数的 C 代码。此外,是否可以有人解释原作者为什么要返回索引/指针?

点赞
用户4117435
用户4117435

newThread将新线程推入Lua堆栈中。没有必要返回一个值。通过指针值进行接口交互在安全性上会有挑战(而且我认为这种方式也很糟糕),而索引可以很容易地通过不返回它(例如通过getTop,通过跟踪堆栈大小或使用负索引)来获得。对于API的所有部分,这都是更多或更少相同的内容,它将值推入堆栈中。

请注意,JNLUA的newThread与Lua的lua_newthread有所不同;JNLUA中的线程要稍微简单一些,更倾向于像Lua协程那样使用(当然,Lua协程也是Lua线程)-newThread将来自堆栈顶部的值用作协程的“开始函数”(当第一次为线程调用resume时将调用它)。

我不会推测作者以这种方式公开线程的决定;返回LuaState(或者公开lua_tothread)也可以实现与原始Lua API更接近的方式。

这是实现的核心:

static int newthread_protected (lua_State *L) {
    lua_State *T;

    T = lua_newthread(L);
    lua_insert(L, 1);
    lua_xmove(L, T, 1);
    return 1;
}

这是在受保护呼叫中完成的,因为lua_newthread可能会引发异常(如果它用完内存)。唯一的参数(因此最初在堆栈上的唯一事物)是索引1处的开始函数。lua_newthread将推入索引2处的新线程对象到堆栈中。然后,lua_insert将翻转新线程对象和开始函数的顺序,而lua_xmove将开始函数转移到新线程,然后返回新线程(这将在调用者的堆栈顶部留下它)。

在Lua中,具有开始函数的新线程可以被lua_resume视为已被yield的线程等效处理,因此这基本上就是JNLUA所做的事情-它的resume可以用于使用先前提供的函数启动线程。

2021-06-06 20:16:30