Lua中一个lua_State就对应一个lua stack,通常情况下一个lua stack就足够了,由此便产生了与C/C++不同的地方:在C/C++中每个函数调用都会有自己对应的stack,在函数调用完成之后,便将其释放掉,因而不需要担心stack会无限制地生长下去;而在C/C++中使用Lua,由于所有的操作都基于同一个stack,因此需要程序员自己在代码中做stack的清理工作。
Lua提供了一对API来帮助程序员操纵stack:lua_gettop和lua_settop。下面给出两种清理栈的方法:
1)为每个操作Lua栈的C/C++函数添加一个栈顶位置参数。在调用该函数时将lua_gettop的结果作为参数传递,在函数调用结束时使用lua_settop以传入的栈顶位置作为参数恢复调用前的栈顶位置。CEGUI中的script module便采用此方法,下面摘录相关代码:
data:image/s3,"s3://crabby-images/d8aef/d8aef1ca72194cc1f263ac1b681faa2e7d2ee4af" alt=""
/**//*************************************************************************
Execute script code string
*************************************************************************/
void LuaScriptModule::executeString(const String& str)
data:image/s3,"s3://crabby-images/d8aef/d8aef1ca72194cc1f263ac1b681faa2e7d2ee4af" alt=""
data:image/s3,"s3://crabby-images/c9e2b/c9e2bc817d66f0a3894ba04ea7703b8e0b7b6162" alt=""
{
int top = lua_gettop(d_state);
data:image/s3,"s3://crabby-images/f74aa/f74aa0daa97912d7a2dcb8fc685747aa4f541b5c" alt=""
executeString_impl(str, initErrorHandlerFunc(), top);
cleanupErrorHandlerFunc();
}
data:image/s3,"s3://crabby-images/54783/547830fede928f19a3ce63b212a632c66666c748" alt=""
void LuaScriptModule::executeString_impl(const String& str, const int err_idx,
const int top)
data:image/s3,"s3://crabby-images/d8aef/d8aef1ca72194cc1f263ac1b681faa2e7d2ee4af" alt=""
data:image/s3,"s3://crabby-images/c9e2b/c9e2bc817d66f0a3894ba04ea7703b8e0b7b6162" alt=""
{
// load code into lua and call it
int error = luaL_loadbuffer(d_state, str.c_str(), str.length(), str.c_str()) ||
lua_pcall(d_state, 0, 0, err_idx);
data:image/s3,"s3://crabby-images/f74aa/f74aa0daa97912d7a2dcb8fc685747aa4f541b5c" alt=""
// handle errors
if (error)
data:image/s3,"s3://crabby-images/788e5/788e5df7a2b54adca27f5032aa9631ef1512545d" alt=""
{
String errMsg = lua_tostring(d_state,-1);
lua_settop(d_state,top);
throw ScriptException("Unable to execute Lua script string: '" +
str + "'\n\n" + errMsg + "\n");
}
data:image/s3,"s3://crabby-images/f74aa/f74aa0daa97912d7a2dcb8fc685747aa4f541b5c" alt=""
lua_settop(d_state,top);
}
2)利用局部变量的创建与销毁来自动实现stack清理。下面为局部变量的类型
struct LuaStackAutoPopup
data:image/s3,"s3://crabby-images/d8aef/d8aef1ca72194cc1f263ac1b681faa2e7d2ee4af" alt=""
{
lua_State* mBackupLS;;
int mPos;
LuaStackAutoPopup( lua_State* vpSL )
data:image/s3,"s3://crabby-images/788e5/788e5df7a2b54adca27f5032aa9631ef1512545d" alt=""
{
mPos = lua_gettop( vpSL );
mBackupLS = vpSL;
}
data:image/s3,"s3://crabby-images/f74aa/f74aa0daa97912d7a2dcb8fc685747aa4f541b5c" alt=""
void Popup( )
data:image/s3,"s3://crabby-images/788e5/788e5df7a2b54adca27f5032aa9631ef1512545d" alt=""
{
lua_settop( mBackupLS,mPos );
mBackupLS = NULL;
}
data:image/s3,"s3://crabby-images/f74aa/f74aa0daa97912d7a2dcb8fc685747aa4f541b5c" alt=""
~LuaStackAutoPopup( )
data:image/s3,"s3://crabby-images/788e5/788e5df7a2b54adca27f5032aa9631ef1512545d" alt=""
{
if( mBackupLS )
lua_settop( mBackupLS,mPos );
}
};
利用此机制,改造方法1中的executeString:
data:image/s3,"s3://crabby-images/d8aef/d8aef1ca72194cc1f263ac1b681faa2e7d2ee4af" alt=""
/**//*************************************************************************
Execute script code string
*************************************************************************/
void LuaScriptModule::executeString(const String& str)
data:image/s3,"s3://crabby-images/d8aef/d8aef1ca72194cc1f263ac1b681faa2e7d2ee4af" alt=""
data:image/s3,"s3://crabby-images/c9e2b/c9e2bc817d66f0a3894ba04ea7703b8e0b7b6162" alt=""
{
LuaStackAutoPopup stack(d_state);
data:image/s3,"s3://crabby-images/f74aa/f74aa0daa97912d7a2dcb8fc685747aa4f541b5c" alt=""
executeString_impl(str, initErrorHandlerFunc());
cleanupErrorHandlerFunc();
}
data:image/s3,"s3://crabby-images/54783/547830fede928f19a3ce63b212a632c66666c748" alt=""
void LuaScriptModule::executeString_impl(const String& str, const int err_idx)
data:image/s3,"s3://crabby-images/d8aef/d8aef1ca72194cc1f263ac1b681faa2e7d2ee4af" alt=""
data:image/s3,"s3://crabby-images/c9e2b/c9e2bc817d66f0a3894ba04ea7703b8e0b7b6162" alt=""
{
// load code into lua and call it
int error = luaL_loadbuffer(d_state, str.c_str(), str.length(), str.c_str()) ||
lua_pcall(d_state, 0, 0, err_idx);
data:image/s3,"s3://crabby-images/f74aa/f74aa0daa97912d7a2dcb8fc685747aa4f541b5c" alt=""
// handle errors
if (error)
data:image/s3,"s3://crabby-images/788e5/788e5df7a2b54adca27f5032aa9631ef1512545d" alt=""
{
String errMsg = lua_tostring(d_state,-1);
throw ScriptException("Unable to execute Lua script string: '" +
str + "'\n\n" + errMsg + "\n");
}
}