为什么std :: string不是模板常量参数的有效类型?

时间:2011-04-27 09:00:07

标签: c++ templates

我正在尝试创建一个用户可以在地图中存储不同类型数据的类。我为bool,int和std :: string创建了一个映射,并创建了模板函数,这样我就不必为每种类型重写get和set函数。

这是我的代码的最小版本:

#include <map>
#include <string>
#include <stdexcept>
#include <iostream>

class Options {
public:
    template<class T>
    void Set(const std::string& name, const T& value) {
        GetMap<T>()[name] = value;
    }
    template<class T>
    T Get(const std::string& name) {
        auto it = GetMap<T>().find(name);
        if(it == GetMap<T>().end()) {
            throw std::runtime_error(name + " not found");
        }
        return it->second;
    }
private:
    std::map<std::string, int> ints_;
    std::map<std::string, std::string> strings_;
    std::map<std::string, bool> bools_;

    template<class T>
    std::map<std::string, T>& GetMap();
    template<bool>
    std::map<std::string, bool>& GetMap() {
        return bools_;
    }
    template<std::string> // error
    std::map<std::string, std::string>& GetMap() {
        return strings_;
    }
    template<int>
    std::map<std::string, int>& GetMap() {
        return ints_;
    }
};

int main() {
    Options o;
    o.Set("test", 1234);
    o.Set<std::string>("test2", "Hello World!");
    std::cout << o.Get<int>("test") << std::endl
              << o.Get<std::string>("test2") << std::endl;
}

我收到以下错误:

error: 'struct std::basic_string<char>' is not a valid type for a template constant parameter

但为什么?

2 个答案:

答案 0 :(得分:7)

如果我理解正确,您正在尝试专门化功能模板GetMap()。但是你的语法不正确;你可能想要:

template<class T>
std::map<std::string, T>& GetMap();

template<>
std::map<std::string, bool>& GetMap<bool>() {
    return bools_;
}

等等。

答案 1 :(得分:6)

两点:

  • 专业化应该在外面类(重要),否则就不会编译
  • 专业化的正确语法如下:

    //outside the class definition
    
    template<>
    std::map<std::string, bool>& Options::GetMap<bool>() {
                               //^^^^^^^^^ dont forget this!
          return bools_;
    }
    template<>
    std::map<std::string, std::string>& Options::GetMap<std::string>() {
                                       //^^^^^^^^^ dont forget this!
          return strings_;
    }
    template<>
    std::map<std::string, int>& Options::GetMap<int>() {
                               //^^^^^^^^^ dont forget this!
          return ints_;
    }