如何使用可变模板来制作一个通用的Lua函数包装器
How to use variadic templates to make a generic Lua function wrapper?
对于我当前的项目,我已经编写了很多C/C++到Lua包装器。其中很多都是简单的setter和getter,所以我设法编写了一些模板,使生成这些模板变得容易,比如:
// Class Return Field
template <typename T, typename U, U T::*Member>
int luaU_get(lua_State* L)
{
T* obj = luaW_check<T>(L, 1);
luaU_push<U>(L, obj->*Member);
return 1;
}
static luaL_reg Foo_Table[] =
{
...
// Now I can just use this generic template to avoid
// writing simple getter functions
{ "getbar", luaU_get<Foo, Bar, &Foo::bar> },
...
};
我也想为任意函数的简单函数包装器做一些类似的事情。例如,能够做到这一点会很好:
template <typename T, typename U, U (T::*Func)(), typename... Args>
int luaU_func(lua_State* L)
{
// ...?
}
static luaL_reg Foo_Table[] =
{
...
{ "baz", luaU_func<Foo, int, &Foo::baz, int, float> },
...
};
这个想法是,当编译时,模板实际上是这样的:
int luaU_func(lua_State* L)
{
luaU_push<int>(L, luaW_check<Foo>(L, 1)->baz(luaU_check<int>(L, 2), luaU_check<float>(L, 3)));
return 1;
}
我只尝试过使用...
扩展器,问题是要将整数值索引映射到正确的参数。我想不出让他们正常工作的方法。这样的事情可能吗?
(这里已经有点神奇了;我为lua_push和lua\heck之类的东西写了一些模板化的包装器。所有现有的包装器都可以在这里找到)
诀窍是通过部分专门化包含包装函数的类模板来利用模板参数推导:
// Lua API dummies ...
struct lua_State {};
template<class T> void luaU_push(lua_State*,T);
template<class T> T* luaW_check(lua_State*,int);
template<class T> T luaU_check(lua_State*,int);
// metaprogramming for creating indices ...
template<int...Ints>
struct int_pack {};
template<int Begin, int Count, int...Tail>
struct make_int_range_type {
typedef typename make_int_range_type<Begin,Count-1,Begin+Count-1,Tail...>::type type;
};
template<int Begin, int...Tail>
struct make_int_range_type<Begin,0,Tail...> {
typedef int_pack<Tail...> type;
};
template<int Begin, int Count>
inline typename make_int_range_type<Begin,Count>::type
make_int_range()
{ return typename make_int_range_type<Begin,Count>::type(); }
// the actual wrapper ...
template<class MemFunPtrType, MemFunPtrType PMF>
struct lua_mem_func_wrapper;
template<class Clazz, class ReturnType, class...Args, ReturnType(Clazz::*PMF)(Args...)>
struct lua_mem_func_wrapper<ReturnType(Clazz::*)(Args...),PMF> {
static int doit(lua_State* L) {
return doit_impl(L,make_int_range<2,sizeof...(Args)>());
}
private:
template<int...Indices>
static int doit_impl(lua_State* L, int_pack<Indices...>) {
luaU_push<ReturnType>(L,
(luaW_check<Clazz>(L, 1)->*PMF)(
luaU_check<Args>(L, Indices)...
)
);
return 1;
}
};
#define GET_MEM_FUN_WRAPPER(...) &lua_mem_func_wrapper<decltype(__VA_ARGS__),__VA_ARGS__>::doit
// testing ...
struct foo {
int baz(int, float);
};
void test() {
auto* ptr = GET_MEM_FUN_WRAPPER(&foo::baz);
}
此代码在G++4.6.1下使用选项-c--std=c++0x进行编译。要查看它是否真的符合您的要求,请测试它。。。
重复使用此答案中的索引生成代码,并忽略对Func
的函数调用(不知道这到底是如何使用的),这就是它的样子:
template <typename T, typename U, U (T::*Func)(),
typename... Args, size_t... Idx>
int luaU_func_impl(lua_State* L, Collection<Idx...>)
{
luaU_push<int>(L, luaW_check<U>(L, 1), luaU_check<Args>(L, Idx+2)...);
return 1;
}
template <typename T, typename U, U (T::*Func)(), typename... Args>
int luaU_func(lua_State* L)
{
typename GenerateCollection<Args...>::type Indices;
return luaU_func_impl<T, U, Func, Args...>(L, Indices);
}
相关文章:
- 如何在C++中将用户数据从一个 Lua 块传递到另一个块
- 是否可以使用另一个lua文件中定义的表,该表在当前文件中不需要作为模块
- 在C 函数中打印一个来自LUA的静态变量
- 如何在LUA中创建一个C++兼容的函数对象
- 而不是在 lua 中创建一个对象,如何让 lua 直接高C++对象来启动方法?
- Lua:为自定义用户数据提供一个字符串方法
- 如何使用可变模板来制作一个通用的Lua函数包装器
- 在Selene中包装一个枚举,以便在Lua中访问
- 将一个没有全局的表从 lua 返回到 C
- C# 应用(通过本机 C++ DLL)与另一个使用嵌入 Lua 的 DLL 的应用之间的 IPC
- 当我为调试Lua编译程序时,运行得很好,但为什么我要为发布版编译它,我得到了一个c0000005错误
- SWIG:Lua-传递一个c++实例作为Lua函数参数
- 在 LuaBridge 中注册一个类的函数,然后通过 lua 脚本调用它?
- 从另一个线程中断c++中的lua脚本
- 使用SWIG封装c++类以在Lua中使用它-需要一个简单的例子
- 注册一个c++类以便在Lua 5.2中使用
- 将表从一个lua状态传递到另一个lua状态
- 编写一个Lua包装器,如何使用可以返回多种类型的方法从堆栈返回一个值
- 从Lua创建一个C++类或"entity"(或等效的东西)
- 使用指针从一个c++对象,到另一个c++对象,修改成员变量,从Lua