我需要检查给定的类是否<<(cls, ostream)
定义了运算符。如果是这样,我希望我的函数使用它来写入ostringstream
,否则应该使用样板代码。
我知道以前有人问过这个问题。但是,我通常会发现并不总是适用于我的编译器 (clang++) 的自定义解决方案。经过几个小时的搜索,我终于找到了 boost::type_traits。我以前没有看过那里,因为我认为 c++11 已经复制了 boost 所具有的特征部门的所有内容。
对我有用的解决方案是:
template <typename C>
std::string toString(C &instance) {
std::ostringstream out;
out << to_string<C, boost::has_left_shift<C, std::ostream>::value>::convert(ctx);
return out.str();
}
to_string
定义为:
template <typename C, bool>
struct to_string {
// will never get called
static std::string convert(LuaContext &ctx) {}
};
template <typename C>
struct to_string<C, true> {
static std::string convert(LuaContext &ctx) {
return "convert(true) called.";
}
};
template <typename C>
struct to_string<C, false> {
static std::string convert(LuaContext &ctx) {
return "convert(false) called.";
}
};
所以我发布这个有两个原因:
检查这是否是最明智的使用方法,或者看看其他人是否可以提出更好的解决方案(即,这个问题更多是出于对方法的好奇,而不是“这行得通吗?”——它已经对我有用了)
发布此内容以节省其他人的搜索时间,以防她/他也需要做类似的事情。
作为一个更普遍的问题——有时特征类似乎返回 std::true_type 或 std::false_type (嗯,至少对于非增强类)。其他时候它们是布尔值。这种差异有原因吗?如果
boost:has_left_shift
返回一个类型而不是一个bool
,那么我可能只有一个to_string
结构。