是否存在(在标准库或Boost中)类型特征来测试类型是否可以表示字符串?
我在使用Boost.Fusion时偶然发现了一个问题:
auto number = fusion::make_vector( 1, "one" );
auto numberName = fusion::filter< char const * >( number );
assert( numberName == fusion::make_vector( "one" ) ); // fails
我希望filter
保留“one”,但它失败了,因为“one”没有衰减到指针(make_vector
通过引用获取其参数,因此类型为const char (&)[4]
)。因此,我需要一个可以让我写下这样的特性:
auto numberName = fusion::filter_if< is_string< mpl::_ > >( number );
我知道char const *
和const char[N]
不一定是以空字符结尾的字符串,但是能够统一检测它们仍然很方便。该特征也可能会返回true
等std::string
。
是否存在这样的特征,还是我必须自己编写?
答案 0 :(得分:6)
我尝试了实现这样的特性,但我不确定它是否真的很强大。任何意见都将不胜感激。
template <typename T>
struct is_string
: public mpl::or_< // is "or_" included in the C++11 library?
std::is_same< char *, typename std::decay< T >::type >,
std::is_same< const char *, typename std::decay< T >::type >
> {};
assert ( ! is_string< int >::value );
assert ( is_string< char * >::value );
assert ( is_string< char const * >::value );
assert ( is_string< char * const >::value );
assert ( is_string< char const * const >::value );
assert ( is_string< char (&)[5] >::value );
assert ( is_string< char const (&)[5] >::value );
// We could add specializations for string classes, e.g.
template <>
struct is_string<std::string> : std::true_type {};
答案 1 :(得分:1)
这应该在C ++ 17中起作用。
#include <iostream>
#include <string>
#include <type_traits>
template<typename T>
struct is_string
: public std::disjunction<
std::is_same<char *, typename std::decay<T>::type>,
std::is_same<const char *, typename std::decay<T>::type>,
std::is_same<std::string, typename std::decay<T>::type>
> {
};
int main()
{
std::cout << std::boolalpha;
std::string str = "i am a string";
std::cout << is_string<decltype(str)>::value << std::endl; // "true"
std::cout << is_string<decltype("i am a string literal")>::value << std::endl; // "true"
}