Lua C API и метатабельные функции - PullRequest
4 голосов
/ 10 августа 2010

Я использую Lua внутри приложения C, и у меня есть две таблицы.Я хочу создать третью таблицу, которая, будучи пустой, будет индексировать значения из моих первых двух таблиц.Я написал следующий простой пример на Lua -

a = { one="1", two="2" }
b = { three="3", four="4" }

meta = { __index = function(t,k)
  if a[k] == nil then return b[k]
  else return a[k] end
end }

c = {}
setmetatable(c, meta)

print(c.one) -- prints "1"
print(c.four) -- prints "4"

Мой вопрос: какой самый эффективный способ сделать это из C API?Я смог сделать это, создав новую таблицу, поместив вышеупомянутый фрагмент кода Lua в эту таблицу, затем вызвав для нее setmetatable (), но это кажется неоптимальным.Есть ли лучший способ?

Ответы [ 2 ]

11 голосов
/ 10 августа 2010
#include <stdio.h>
#include "lua.h"

/* __index metamethod for the 'c' table (stack: 1 = table 'c', 2 = desired index) */
static int
cindex(lua_State *L)
{
    /* try the global 'a' table */
    lua_getglobal(L, "a");
    lua_pushvalue(L, 2);
    lua_gettable(L, -2);
    if (!lua_isnil(L, -1))
        return 1;

    /* try the global 'b' table */
    lua_getglobal(L, "b");
    lua_pushvalue(L, 2);
    lua_gettable(L, -2);
    if (!lua_isnil(L, -1))
        return 1;

    /* return nil */
    return 0;
}

int
main(int argc, char **argv)
{
    lua_State *L;

    L = (lua_State *) luaL_newstate();
    luaL_openlibs(L);

    /* create the global 'a' table */
    lua_createtable(L, 0, 2);
    lua_pushstring(L, "1");
    lua_setfield(L, -2, "one");
    lua_pushstring(L, "2");
    lua_setfield(L, -2, "two");
    lua_setglobal(L, "a");

    /* create the global 'b' table */
    lua_createtable(L, 0, 2);
    lua_pushstring(L, "3");
    lua_setfield(L, -2, "three");
    lua_pushstring(L, "4");
    lua_setfield(L, -2, "four");
    lua_setglobal(L, "b");

    /* create the global 'c' table and use a C function as the __index metamethod */
    lua_createtable(L, 0, 0);
    lua_createtable(L, 0, 1);
    lua_pushcfunction(L, cindex);
    lua_setfield(L, -2, "__index");
    lua_setmetatable(L, -2);
    lua_setglobal(L, "c");

    /* run the test script */
    luaL_loadstring(L, "print(c.one)\nprint(c.four)");
    if (0 != lua_pcall(L, 0, 0, 0)) {
        puts(lua_tostring(L, -1));
        return 1;
    }

    return 0;
}
2 голосов
/ 10 августа 2010

Можете ли вы изменить метатабельность b?Если это так, это более эффективно:

a = { one="1", two="2" }
b = { three="3", four="4" }

setmetatable(b, { __index = a })

-- setmetatable(x, m) returns x, so you can do this:
c = setmetatable({}, { __index = b}) -- meta is here, too

print(c.one) -- prints "1"
print(c.four) -- prints "4"

Когда __index указывает на таблицу, это более эффективно, чем когда оно указывает на функцию;Я где-то читал, что это было эквивалентно 3 косвенным обращениям в C. Так что в худшем случае (c.one) всего 6 косвенных указаний.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...