2

在使用绑定实现调度表的过程中,我试图用函数模板替换宏

一旦我开始添加用于返回的表格std::stringdouble.

宏版本工作正常,但模板版本转储核心。

有人可以解释我做错了什么吗?谢谢你。

代码

#include <functional>
#include <iostream>
#include <map>

struct Integer
{
    virtual int getInt() const = 0;
};

struct IntImpl : public Integer
{
    virtual int getInt() const { return 42; }
};

typedef std::function<int()>                               IntFunction;
typedef std::function<IntFunction( Integer const& inst )>  IntLambda;

#define USE_MACRO

#ifdef USE_MACRO
#define MP(A,B) \
    std::make_pair( A, []( Integer const& inst ) { \
                return std::bind( B, std::cref( inst )); \
            } )
#else
template<typename L,typename T,typename M>
std::pair<std::string,L>
MP( std::string const& str, M method)
{
    return std::make_pair( str, [&method]( T const& inst ) {
        return std::bind( method, std::cref( inst ));
        } 
    );
}
#endif

static std::map<std::string,IntLambda> const g_intTbl =
{
#ifdef USE_MACRO
    MP( "getInt", &Integer::getInt )
#else
    MP<IntLambda,Integer>( "getInt", &Integer::getInt )
#endif
};

int
main( int argv, char* argc[] )
{
    IntImpl x;
    std::cerr << g_intTbl.find("getInt")->second( x )() << std::endl;
}
4

1 回答 1

4

您的问题在于如何捕获method到您的 lambda 中。您正在method通过引用捕获,它引用本地方法参数,堆栈上的一个值。在return std::bind()调用函数之前不会执行,此时它将尝试将引用绑定到堆栈变量,该变量显然不再存在。

您只需要更改[&method][method]按值捕获。

于 2013-01-07T01:09:01.290 回答