开发人员常常需要解析字符串。你可以使用低层的C函数strtok(尽管不推荐这样做),或者你可以自己写一个字符串分割器(string tokenizer)或者找一个现成的实现。
不管怎样,你通常会需要作一些基本的解析,其中分隔字符(token)用的是空格字符。对于这个基本需求,你可以从你的字符串创建一个stringstream,并用运算符“〉〉”来解析出每个词(然后把它视作你希望的任何意义)。
下面是一个例子:
#include 〈string〉
#include 〈iostream〉
#include 〈sstream〉
int main(int argc, char* argv[])
{
// 输入字符串
std::string strInput = "John Doe 355223";
// 创建一个stringstream,逐个单词地解析它
std::stringstream streamIn( strInput);
std::string strFirstName;
std::string strLastName;
int nID;
streamIn 〉〉 strFirstName 〉〉 strLastName 〉〉 nID;
// 显示我们解析出来的信息
std::cout 〈〈 "First name: " 〈〈 strFirstName 〈〈 std::endl
〈〈 "Last name: " 〈〈 strLastName 〈〈 std::endl
〈〈 "Employee ID: " 〈〈 nID 〈〈 std::endl;
return 0;
}
你可以自动化上面的过程,通过一个函数,只要给它一个字符串作为参数,就返回一个stringstream供你稍后解析
to_stream函数的代码并不像你想象的那么直截了当。这是因为我们要返回一个临时值,而这个值稍后会被视作一个常量(constant)。你不能写入到一个常量stringstream。因此,我们需要给stringstream加一层包装。
#include 〈iostream〉
#include 〈string〉
#include 〈sstream〉
namespace Private
{
template〈 class char_type, class char_traits〉
struct stream_holder
{
typedef stream_holder〈 char_type, char_traits〉 this_class;
typedef std::basic_istringstream〈 char_type, char_traits〉 stream_type;
typedef std::basic_string〈 char_type, char_traits〉 string_type;
stream_holder( const string_type & value)
: m_stream( value)
{}
stream_holder( const this_class & source)
: m_stream( source.m_stream.str() )
{}
// 允许将这个stream传给接受stream作为参数的函数
operator stream_type & () const
{ return m_stream; }
private:
mutable stream_type m_stream;
};
template〈 class char_type, class char_traits, class value_type〉
inline typename stream_holder〈 char_type, char_traits〉::stream_type & operator 〉〉 (const stream_holder〈 char_type, char_traits〉 & streamIn, value_type & value)
{
typedef typename stream_holder〈 char_type, char_traits〉::stream_type stream_type;
stream_type & underlyingStream = streamIn;
underlyingStream 〉〉 value;
return underlyingStream;
}
} // namespace Private
template〈 class char_type, class char_traits〉
Private::stream_holder〈 char_type, char_traits〉 to_stream(
const std::basic_string〈 char_type, char_traits〉 & str)
{
return str;
}
template〈 class char_type〉
Private::stream_holder〈 char_type, std::char_traits〈 char_type〉 〉 to_stream( const char_type * str)
{
return ( std::basic_string〈 char_type〉)str;
}
现在,前面的例子变成这个样子:
int main(int argc, char* argv[])
{
std::string strFirstName;
std::string strLastName;
int nID;
to_stream( "John Doe 355223") 〉〉 strFirstName 〉〉 strLastName 〉〉 nID;
// 显示我们解析出来的信息
std::cout 〈〈 "First name: " 〈〈 strFirstName 〈〈 std::endl
〈〈 "Last name: " 〈〈 strLastName 〈〈 std::endl
〈〈 "Employee ID: " 〈〈 nID 〈〈 std::endl;
return 0;
}