struct PictureCfg
{
string name;
float rotation;
};
1,全局配置
-- picture0-cfg.lua
name ="dragon.png"
rotation= 180
// 全局配置的读取
voidLoadPictureCfg(vector<PictureCfg>& rstPictureCfgVec)
{
PictureCfg stCfg;
lua_State* L = luaL_newstate(); // 创建的一个新的独立的状态机
luaL_dofile(L, "picture0-cfg.lua");
lua_getglobal(L, "name"); // 把全局变量 name 里的值压入堆栈,等价于lua_getfield(L,LUA_GLOBALSINDEX, "name")
printf("%s\n", lua_typename(L, lua_type(L, -1)));
assert(lua_isstring(L, -1) == 1); // 当给定索引的值是一个字符串或是一个数字(数字总能转换成字符串)时,返回 1 ,否则返回 0
stCfg.name = lua_tostring(L, -1); // 把给定索引处的 Lua 值转换为一个 C 字符串,注意该系列的函数不会将值从栈上弹出的
lua_getglobal(L, "rotation"); //
assert(lua_isnumber(L, -1) == 1);
stCfg.rotation = (float) lua_tonumber(L, -1); // 把给定索引处的 Lua 值转换为lua_Number 这样一个 C 类型(参见 lua_Number )。这个 Lua 值必须是一个数字或是一个可转换为数字的字符串(参见§2.2.1 );否则,lua_tonumber 返回 0 。typedef double lua_Number;
rstPictureCfgVec.push_back(stCfg);
lua_close(L);
}
2,一维表配置的读取
-- picture1-cfg.lua
cfg = { name="dragon.png",rotation=180 }
// 一维表的读取
voidLoadPictureCfg(vector<PictureCfg>& rstPictureCfgVec)
{
lua_State* L = luaL_newstate();
luaL_dofile(L, "picture1-cfg.lua");
// get the table
lua_getglobal(L, "cfg" ); // 将全局变量cfg压栈,cfg是table类型的,一维表
assert(lua_istable(L, -1) == 1);
PictureCfg stCfg;
// push the key to stack for getting the value
lua_pushstring(L, "name"); //
// now the table is in the -2 and key in the top(-1)
lua_gettable(L, -2);
/*
void lua_gettable (lua_State *L, int index);
把 t[k] 值压入堆栈,这里的 t 是指有效索引 index 指向的值,而 k 则是栈顶放的值。
这个函数会弹出堆栈上的key (把结果放在栈上相同位置)。
*/
assert(lua_isstring(L, -1 ));
stCfg.name = lua_tostring(L, -1 );
lua_pop(L, 1 );
// push the key to stack for getting the value
lua_pushstring(L, "rotation");
// now the table is in the -2 and key in the top(-1)
lua_gettable(L, -2 );
assert(lua_isnumber(L, -1));
stCfg.rotation = lua_tonumber(L, -1);
rstPictureCfgVec.push_back(stCfg);
/* 清空栈 */
lua_pop(L, 2);
lua_close(L);
}
典型的一维表遍历方法是这样的:
int t_idx = lua_gettop(L); // 取 table 索引值
lua_pushnil(L); // 第一个 key
while (lua_next(L, t_idx) != 0)
{
// 现在栈顶(-1)是 value,-2 位置是对应的 key
// 这里可以判断 key 是什么并且对 value 进行各种处理
printf("%s - %s\n",lua_typename(L, lua_type(L, -2)), lua_typename(L, lua_type(L, -1)));
lua_pop(L, 1); // 移除 'value' ;保留 'key' 做下一次迭代
}
2,二维表配置的读取
-- picture2-cfg.lua
cfg = {
{ name="dragon.png" ,rotation=180 } ,
{ name="dragon.png" ,rotation=0 }
}
// 二维表的读取
voidLoadPictureCfg(vector<PictureCfg>& rstPictureCfgVec)
{
lua_State* L = luaL_newstate();
luaL_dofile(L, "picture2-cfg.lua");
// get the table
lua_getglobal(L, "cfg"); // 将全局变量cfg压栈,cfg是table类型的,且是二维表
assert(lua_istable(L, -1) == 1);
/*
int lua_type (lua_State *L, int index);
返回给定索引处的值的类型,当索引无效时则返回 LUA_TNONE (那是指一个指向堆栈上的空位置的索引)。 lua_type 返回的类型是一些个在 lua.h 中定义的常量: LUA_TNIL , LUA_TNUMBER , LUA_TBOOLEAN , LUA_TSTRING , LUA_TTABLE , LUA_TFUNCTION , LUA_TUSERDATA , LUA_TTHREAD , LUA_TLIGHTUSERDATA 。
const char *lua_typename (lua_State*L, int tp);
返回 tp 表示的类型名,这个 tp 必须是 lua_type 可能返回的值中之一。
*/
lua_pushnil(L); /* first key */ // 把一个 nil 压栈作为遍历table的第一个key
while (lua_next(L, -2) != 0) // index -2 is the table
{
/*
int lua_next (lua_State *L, int index);
从栈上弹出一个 key(键),然后把index指定的表中的下一个key-value(健值)对压入堆栈
如果表中已无更多元素,那么 lua_next 将返回 0 (什么也不压入堆栈)。
*/
/* 'key' (at index -2) and 'value' (at index -1) */
PictureCfg rstCfg;
// push the key to stack for getting the value
lua_pushstring(L, "name"); //
/*
void lua_pushstring (lua_State *L, const char *s);
把指针 s 指向的以零结尾的字符串压栈。 Lua 对这个字符串做一次内存拷贝(或是复用一个拷贝),因此 s 处的内存在函数返回后,可以释放掉或是重用于其它用途。字符串中不能包含有零字符;第一个碰到的零字符会认为是字符串的结束。
*/
// now the table is in the -2 and key in the top(-1)
lua_gettable(L, -2);
/*
void lua_gettable (lua_State *L, int index);
把 t[k] 值压入堆栈,这里的 t 是指有效索引 index 指向的值,而 k 则是栈顶放的值。
这个函数会弹出堆栈上的 key (把结果放在栈上相同位置)。
*/
assert(lua_isstring(L, -1)); // 第一个子表的name的值现在位于栈顶
rstCfg.name = lua_tostring(L, -1);
lua_pop(L, 1);
/*
void lua_pop (lua_State *L, int n);
从堆栈中弹出 n 个元素。
*/
// push the key to stack for getting the value
lua_pushstring(L, "rotation");
// now the table is in the -2 and key in the top(-1)
lua_gettable(L, -2);
assert(lua_isnumber(L, -1));
rstCfg.rotation = lua_tonumber(L, -1);
rstPictureCfgVec.push_back(rstCfg);
/* removes the key we pushed and the 'value' of the global table ; keeps 'key' for next iteration */
lua_pop(L, 2);
}
lua_close(L);
}
另一种遍历二维表的方式如下:
t_idx = lua_gettop(L);
lua_pushnil(L);
while (lua_next(L, t_idx))
{
it_idx = lua_gettop(L); // inner table
lua_pushnil(L);
while(lua_next(L, it_idx))
{
printf("%s\n", lua_tostring(L,-1)); // 这里是取value,但value不一定为string类型的
lua_pop(L, 1);
}
lua_pop(L, 1);
}