用std::pair数组初始化std::map问题(指针错误?)



我试图理解如何正确初始化私有const std::map 。我研究了这个众所周知的话题,但没有一个答案适合我,因为我被迫使用旧的gcc和boost版本(gcc 4.4, boost 1.41 -> c++ 11功能是有限的,boost::asign::map_list_of不编译),而且,我不想使用类成员函数std::map初始化,因为需要在对象构造时在类之外这样做。

然而,std::map有一个接受两个迭代器的重载构造函数(这个线程也启发了我),这是我的解决方案:

template <typename Arg, typename Callback>
class CallbackSelector
{
    private:
        const std::map<Arg, Callback> Mapping;
    public:
        CallbackSelector(std::pair<Arg, Callback> _Mapping[]):
            Mapping(_Mapping, _Mapping + sizeof(_Mapping)/sizeof(std::pair<Arg, Callback>))
            {
                //BOOST_ASSERT(sizeof _Mapping)/(sizeof (std::pair<Arg, Callback>)) == 2);
                std::cout << "sizeof _Mapping " << sizeof _Mapping << std::endl;
                std::cout << "sizeof (std::pair<Arg, Callback>) " << sizeof (std::pair<std::string, boost::function<void(void)> >) << std::endl;
            };
};
void PamEvent() {}
void DefaultEvent() {}
int main(int argc, char** argv)
{
    std::pair<std::string, boost::function<void(void)> > _Mapping[] =
    {
        std::make_pair("pam:", boost::bind(&PamEvent)),
        std::make_pair("none", boost::bind(&DefaultEvent))
    };
    std::cout << "sizeof _Mapping " << sizeof _Mapping << std::endl;
    std::cout << "sizeof (std::pair<Arg, Callback>) " << sizeof (std::pair<std::string, boost::function<void(void)> >) << std::endl;
    CallbackSelector<std::string, boost::function<void(void)> > Selector(_Mapping);
}

(此示例的完整可执行版本)

如果我取消BOOST_ASSERT宏的注释行,这段代码将不会被编译,因为映射的大小不等于2,这是由于将std::pair<std::string, boost::function<void(void)> > _Mapping[]传递到CallbackSelector类构造函数的错误。您可以通过查看输出来验证:

sizeof _Mapping 80                                 //in the int main()
sizeof (std::pair<Arg, Callback>) 40
sizeof _Mapping 8                                  //in the CallbackSelector constructor
sizeof (std::pair<Arg, Callback>) 40               //the array size has decreased tenfold
如果有人能发现这个(显然?)简单的错误,我会很高兴。谢谢你。

没有办法确定数组的大小,只给它一个指针。您可以将大小作为单独的参数传递:

CallbackSelector(std::pair<Arg, Callback> mapping[], size_t size) :
    Mapping(mapping, mapping + size)

或将其推断为模板形参:

template <size_t size>
CallbackSelector(std::pair<Arg, Callback> (&mapping)[size]) :
    Mapping(mapping, mapping + size)

或者,在c++ 11或更高版本中,您可以使用initializer_list参数

CallbackSelector(std::initializer_list<std::pair<Arg, Callback>> mapping) :
    Mapping(mapping.begin(), mapping.end())
// Usage example
CallbackSelector<std::string, boost::function<void(void)> > selector {
    {arg1, callback1}, 
    {arg2, callback2}
};

(注意:我冒昧地重命名了_Mapping,这是一个您不应该使用的保留名称)

相关内容

  • 没有找到相关文章

最新更新