模板变量作为字符串而不是const char*

Dil*_*rix 5 c++ string templates char

我喜欢矢量,通常在数组上使用它们.出于这个原因,我创建了一个模板化的可变参数函数来初始化向量(包含在下面).

标题(.h):

template <typename T>
vector<T> initVector(const int argCount, T first, ...);
Run Code Online (Sandbox Code Playgroud)

来源(.hpp):

template <typename T>
vector<T> initVector(const int argCount, T first, ...) {
    vector<T> retVec;
    retVec.resize(argCount);

    if(argCount < 1) { ... }

    retVec[0] = first;

    va_list valist;
    va_start(valist, first);
    for(int i = 0; i < argCount-1; i++) { retVec[i+1] = va_arg(valist, T); }
    va_end(valist);

    return retVec;
}
Run Code Online (Sandbox Code Playgroud)

它适用于大多数类型(例如int,double ...),但不适用于字符串---因为编译器将它们解释为'const char*',因此

vector<string> strvec = initVector(2, "string one", "string two");
Run Code Online (Sandbox Code Playgroud)

给我错误:

error: conversion from ‘std::vector<const char*, std::allocator<const char*> >’ to non-scalar type ‘std::vector<std::basic_string<char, std::char_traits<char>, std::allocator<char> >, std::allocator<std::basic_string<char, std::char_traits<char>, std::allocator<char> > > >’ requested
Run Code Online (Sandbox Code Playgroud)

是否有任何方法可以将字符串参数解释为字符串,而无需每个参数?

orl*_*rlp 4

因为常量的类型"string one"const char*和不是,std::string所以需要进行转换。va_arg无法进行此转换,因此我们需要第二个模板参数:

template <typename VecT, typename EleT>
std::vector<VecT> init_vector(const size_t nargs, EleT first, ...) {
    std::vector<VecT> result;
    result.reserve(nargs);

    if (nargs == 0) {
        return result;
    }

    result.push_back(first);

    if (nargs == 1) {
        return result;
    }

    va_list valist;
    va_start(valist, first);

    for (int i = 1; i < nargs; ++i) {
        result.push_back(VecT(va_arg(valist, EleT)));
    }

    va_end(valist);

    return result;
}

std::vector<std::string> = init_vector<std::string>(2, "string one", "string two")
Run Code Online (Sandbox Code Playgroud)

请注意,我做了一些更改,最明显的是更改resizereserve,以防止创建不必要的对象。


您也可以简单地使用它(没有元素数量混乱的风险,并且类型安全):

const char *args[] = {"string one" , "string two"};
std::vector<std::string> strvec(args, args + sizeof(args)/sizeof(args[0]))
Run Code Online (Sandbox Code Playgroud)

或者使用 C++11 初始值设定项列表:

std::vector<std::string> strvec = {"string one" , "string two"};
Run Code Online (Sandbox Code Playgroud)

为了好玩,我做了一个更整洁、更安全的小东西,但不会概括为任意数量的参数。它通过超载来工作。以下是前三个重载和示例用法:

template<class C>
inline C init_container() {
    return C();
}

template<class C, class T>
inline C init_container(T arg0) {
    const T args[1] = {arg0};
    return C(args, args + 1);
}

template<class C, class T>
inline C init_container(T arg0, T arg1) {
    const T args[2] = {arg0, arg1};
    return C(args, args + 2);
}

std::vector<std::string> vec =
    init_container< std::vector<std::string> >("hello", "world");
Run Code Online (Sandbox Code Playgroud)

完整的标头(最多 100 个参数)可以在此处下载: https: //gist.github.com/3419369