use*_*225 3 c++ lua memory-management lua-userdata
我怎样才能释放分配的内存lua_newuserdata?
我有一个名为类Foo,这个类有一个构造函数和desstructor,我需要执行两个,但我不知道如何使用C++运算符delete,因为我没有使用new分配内存.
我尝试在创建对象的Lua函数new中执行此操作:
Foo *pf = reinterpret_cast<Foo *>(
                lua_newuserdata(L, sizeof(Foo)));
并在gc函数我试过:
Foo *foo = reinterpret_cast<Foo *>(lua_touserdata(L, 1));
delete foo;
但是我遇到了分段错误.
在这种情况下,您需要使用名为userdatum的lua概念,这意味着,您需要使用lua_newuserdata为您的对象分配指针.
分配内存做类似的事情:
Foo **pfoo = reinterpret_cast<Foo **>(lua_newuserdata(L, sizeof(Foo*)));
*pfoo = new Foo(foo);
在垃圾收集器功能,你可以这样做:
Foo **foo = reinterpret_cast<Foo **>(lua_touserdata(L, 1));
delete *foo;
下面是使用userdatum概念的完整代码示例
#define FOO     "foo"
class Foo {
public:
  Foo(const char *name) {
    this->name = (char *) malloc(strlen(name) + 1);
    strncpy(this->name, name, strlen(name));
  }
  Foo(const Foo &obj) {
    this->name = (char *) malloc(strlen(name) + 1);
    strncpy(this->name, obj.name, strlen(obj.name));
  }
  const char* get_name() const {
    return this->name;
  }
  ~Foo() {
    free(this->name);
  }
private:
  char *name;
};
static Foo* push_foo(lua_State *L, Foo foo) {
  Foo **pfoo = reinterpret_cast<Foo **>(
                lua_newuserdata(L, sizeof(Foo*)));
  *pfoo = new Foo(foo);
  luaL_getmetatable(L, FOO);
  lua_setmetatable(L, -2);
  return *pfoo;
}
static Foo* chk_foo(lua_State *L, int index) {
  Foo *foo;
  luaL_checktype(L, index, LUA_TUSERDATA);
  foo = *reinterpret_cast<Foo **>(luaL_checkudata(L, index, FOO));
  if (foo == NULL)
    luaL_error(L, "error");
  return foo;
}
static int foo_new(lua_State *L) {
  int argc = lua_gettop(L);
  if(argc != 1)
    luaL_error(L, "string argument expected");
  const char* str = luaL_checkstring(L, 1);
  push_foo(L, Foo(str));
  luaL_getmetatable(L, FOO);
  lua_setmetatable(L, -2);
  std::cout << "Lua object created!" << std::endl;
  return 1;
}
static int foo_get(lua_State *L) {
  Foo *foo = chk_foo(L, 1);
  luaL_argcheck(L, foo != NULL, 1, "Error foo");
  lua_pushstring(L, foo->get_name());
  return 1;
}
static int foo_gc(lua_State *L) {
  Foo **foo = reinterpret_cast<Foo **>(lua_touserdata(L, 1));
  luaL_argcheck(L, *foo != NULL, 1, "Error foo");
  delete *foo;
  std::cout << "Lua GC executed!" << std::endl;
  return 0;
}
int luaopen_foolib(lua_State *L) {
  static const luaL_Reg Obj_lib[] = {
    { "get", &foo_get },
    { NULL, NULL }
  };
  static const luaL_Reg LuaLib_Foo[] = {
    { "new", &foo_new },
    { NULL, NULL }
  };
  luaL_newlib(L, LuaLib_Foo);
  // Stack: MyLib
  luaL_newmetatable(L, FOO);
  luaL_newlib(L, Obj_lib);
  lua_setfield(L, -2, "__index");
  lua_pushstring(L, "__gc");
  lua_pushcfunction(L, foo_gc);
  lua_settable(L, -3);
  lua_pop(L, 1);
  return 1;
}
int main(int argc, char **argv) {
  lua_State *L;
  L = luaL_newstate();
  luaL_openlibs(L);
  luaL_requiref(L, "foo", &luaopen_foolib, 1);
  lua_pop(L, 1);
  const char *code = "f = foo.new(\"my_test\")\nprint(f:get())";
  if(luaL_loadstring(L, code) != 0)
  {
    std::cout << "Could not load: " << argv[1] << std::endl;
    exit(EXIT_FAILURE);
  }
  if(lua_pcall(L, 0, 0, 0) != 0)
  {
    std::cout << "Error: " << lua_tostring(L, -1) << std::endl;
    lua_pop(L, 1);
    exit(EXIT_FAILURE);
  }
  lua_close(L);
  return 0;
}
| 归档时间: | 
 | 
| 查看次数: | 1495 次 | 
| 最近记录: |