【发布时间】:2020-12-28 12:41:04
【问题描述】:
我正在尝试遍历 std::tuple 并检索它占用的真实空间量(字节)。要处理 std::string 我需要使用 .length() 函数,但我无法编译,并且出现以下错误:
错误 C2440 '':无法从 'char' 转换为 'std::string'
我也尝试过使用std::is_same(如评论)但没有成功。函数iterate_tuple_types 是这样调用的:
tuple <char, unsigned int, double, std::string> src;
src = make_tuple('a', 10, 0.333333333333333, std::string("some string"));
iterate_tuple_types(src);
代码
template <typename T, typename S>
struct is_string {
static const bool value = false;
};
template <class T, class Traits, class Alloc>
struct is_string<T, std::basic_string<T, Traits, Alloc> > {
static const bool value = true;
};
template<int N, typename... Ts> using NthTypeOf =
typename std::tuple_element<N, std::tuple<Ts...>>::type;
template <size_t I = 0, typename... Ts>
typename std::enable_if<I == sizeof...(Ts), size_t>::type
iterate_tuple_types(std::tuple<Ts...> tup) {
return 0;
}
template <size_t I = 0, typename... Ts>
typename std::enable_if<(I < sizeof...(Ts)), size_t>::type
iterate_tuple_types(std::tuple<Ts...> tup) {
size_t result = 0;
// std::is_same<std::string, NthTypeOf<I, Ts...> >::value
if (std::is_integral<NthTypeOf<I, Ts...> >::value) {
result += sizeof(NthTypeOf<I, Ts...>);
}
else if (is_string<std::string, NthTypeOf<I, Ts...> >::value) {
result += static_cast<size_t>(std::string(std::get<I>(tup)).length());
}
return result + iterate_tuple_types<I + 1>(tup);
}
我这样做对吗?为什么会出现错误?
编辑
我能够找到如下所示的字符串的长度,但是构造一个字符串流并将内容刷新到它似乎效率很低?
typename std::tuple_element<I, std::tuple<Ts...> >::type test = std::get<I>(tup);
std::stringstream ss(test);
result += static_cast<size_t>(ss.str().length());
【问题讨论】:
-
is_string<std::string, std::string>::value是false,因为std::string不是std::basic_string<std::string>。即使它是真的——你正在做一个运行时检查,在运行时没有采用的分支仍然需要编译。