为什么编译器不会停止在此模板代码上给出错误

Why compiler will not stop giving errors on this template code

本文关键字:代码 错误 出错 编译器 为什么      更新时间:2023-10-16

我试图理解为什么这段代码会永久生成编译错误,据我所知,模板专用化应该防止模板在达到 INDEX = 0 时进一步扩展,这应该是代码中唯一的递归。

template <typename STREAM, typename TUPLE, std::size_t INDEX> struct streamer {
STREAM &operator()(STREAM &out, const TUPLE &tuple) {
streamer<STREAM, TUPLE, INDEX - 1U> stream;
return out << stream(out, tuple) << std::get<INDEX>(tuple);
}
};
template <typename STREAM, typename TUPLE> struct streamer<0> {
STREAM &operator()(STREAM &out, const TUPLE &tuple) {
return out << std::get<INDEX>(tuple);
}
};
template <typename STREAM, typename TUPLE>
STREAM &operator<<(STREAM &out, const TUPLE &tuple) {
streamer<STREAM, TUPLE, std::tuple_size<TUPLE>::value - 1U> stream;
return stream(out, tuple);
}
int main() {
std::tuple<std::string, std::string> a_guy("name", "surname");
std::cout << a_guy << std::endl;
return 0;
}

请注意,问题不是关于"如何打印元组",因为已经有很多。问题是关于为什么这个特定的代码不起作用,特别是为什么它使编译器永远循环。

你的专业化有语法错误,这是编译器告诉你的。此外,您的专业化中没有名为INDEX的参数,这是另一个错误。正确的语法应该是:

template <typename STREAM, typename TUPLE> struct streamer<STREAM, TUPLE, 0> {
//                                                         ^^^^^^  ^^^^^  
STREAM &operator()(STREAM &out, const TUPLE &tuple) {
return out << std::get<0>(tuple);
//                        ^^^
}
};

您的代码还有另一个问题:

return out << stream(out, tuple) << std::get<INDEX>(tuple);

这调用operator<<(std::ostream&, std::ostream&)不存在,但无论如何都不需要。只需使用return stream(out, tuple) << std::get<INDEX>(tuple);.