sstream标准库

C++的sstream标准库介绍

我们看一下C++风格的串流控制,C++引入了ostringstream、istringstream、stringstream这三个类,要使用他们创建对象就必须包含sstream.h头文件。 

  istringstream类用于执行C++风格的串流的输入操作。 

  ostringstream类用于执行C风格的串流的输出操作。 

  stringstream类同时可以支持C风格的串流的输入输出操作。

  istringstream类是从istream(输入流类)和stringstreambase(c++字符串流基类)派生而来, ostringstream是从ostream(输出流类)和stringstreambase(c++字符串流基类)派生而来, stringstream则是从iostream(输入输出流类)和和stringstreambase(c++字符串流基类)派生而来。 

  他们的继承关系如下图所示:

 

  istringstream是由一个string对象构造而来,istringstream类从一个string对象读取字符。 

  istringstream的构造函数原形如下: 

  istringstream::istringstream(string str); 

//所有稿件均有版权,如要转载,请务必著名出处和作者  

#include   

#include   

using namespace std;  

int main()   

{  

istringstream istr;  

istr.str("1 56.7",);  

//上述两个过程可以简单写成 istringstream istr("1 56.7"); 

cout << istr.str()<

int a;  

float b;  

istr>>a;  

cout<

istr>>b;  

cout<

system("pause");  

  上例中,构造字符串流的时候,空格会成为字符串参数的内部分界,例子中对a,b对象的输入"赋值"操作证明了这一点,字符串的空格成为了整型数据与浮点型数据的分解点,利用分界获取的方法我们事实上完成了字符串到整型对象与浮点型对象的拆分转换过程。 

  str()成员函数的使用可以让istringstream对象返回一个string字符串(例如本例中的输出操作(cout<

  ostringstream同样是由一个string对象构造而来,ostringstream类向一个string插入字符。 

  ostringstream的构造函数原形如下: 

  ostringstream::ostringstream(string str); 

  示例代码如下:

//程序作者:管宁  

//站点:www.cndev-lab.com  

//所有稿件均有版权,如要转载,请务必著名出处和作者  

#include   

#include   

#include   

using namespace std;  

int main()   

{  

ostringstream ostr;  

//ostr.str("abc");//如果构造的时候设置了字符串参数,那么增长操作的时候不会从结尾开始增加,而是修改原有数据,超出的部分增长 

ostr.put('d');  

ostr.put('e');  

ostr<<"fg";  

  

string gstr = ostr.str();  

cout<

system("pause");  

}

  在上例代码中,我们通过put()或者左移操作符可以不断向ostr插入单个字符或者是字符串,通过str()函数返回增长过后的完整字符串数据,但值得注意的一点是,当构造的时候对象内已经存在字符串数据的时候,那么增长操作的时候不会从结尾开始增加,而是修改原有数据,超出的部分增长。

  对于stringstream了来说,不用我多说,大家也已经知道它是用于C++风格的字符串的输入输出的。 

  stringstream的构造函数原形如下: 

  stringstream::stringstream(string str);

  示例代码如下:

//程序作者:管宁  

//站点:www.cndev-lab.com  

//所有稿件均有版权,如要转载,请务必著名出处和作者  

#include   

#include   

#include   

using namespace std;  

  

int main()   

{  

    stringstream ostr("ccc");  

    ostr.put('d');  

    ostr.put('e');  

    ostr<<"fg";  

    string gstr = ostr.str();  

    cout<

  

    char a;  

    ostr>>a;  

    cout<

      

    system("pause");  

}

  除此而外,stringstream类的对象我们还常用它进行string与各种内置类型数据之间的转换。 

  示例代码如下:

//程序作者:管宁  

//站点:www.cndev-lab.com  

//所有稿件均有版权,如要转载,请务必著名出处和作者  

#include   

#include   

#include   

using namespace std;  

  

int main()   

{  

    stringstream sstr;  

//--------int转string----------- 

    int a=100;  

    string str;  

    sstr<

    sstr>>str;  

    cout<

//--------string转char[]-------- 

    sstr.clear();//如果你想通过使用同一stringstream对象实现多种类型的转换,请注意在每一次转换之后都必须调用clear()成员函数。 

    string name = "colinguan";  

    char cname[200];  

    sstr<

    sstr>>cname;  

    cout<

    system("pause");  

}

  接下来我们来学习一下输入/输出的状态标志的相关知识,C++中负责的输入/输出的系统包括了关于每一个输入/输出操作的结果的记录信息。这些当前的状态信息被包含在io_state类型的对象中。io_state是一个枚举类型(就像open_mode一样),以下便是它包含的值。 

goodbit 无错误 

Eofbit 已到达文件尾 

failbit 非致命的输入/输出错误,可挽回 

badbit 致命的输入/输出错误,无法挽回 

  有两种方法可以获得输入/输出的状态信息。一种方法是通过调用rdstate()函数,它将返回当前状态的错误标记。例如,假如没有任何错误,则rdstate()会返回goodbit. 

  下例示例,表示出了rdstate()的用法: 

//程序作者:管宁    

//站点:www.cndev-lab.com    

//所有稿件均有版权,如要转载,请务必著名出处和作者    

  

#include   

using namespace std;  

  

int main()   

{  

    int a;  

    cin>>a;  

    cout<

    if(cin.rdstate() == ios::goodbit)  

    {  

        cout<<"输入数据的类型正确,无错误!"<

    }  

    if(cin.rdstate() == ios_base::failbit)  

    {  

        cout<<"输入数据类型错误,非致命错误,可清除输入缓冲区挽回!"<

    }  

    system("pause");  

}

  另一种方法则是使用下面任何一个函数来检测相应的输入/输出状态:

bool bad(); 

bool eof(); 

bool fail(); 

bool good(); 

  下例示例,表示出了上面各成员函数的用法:

//程序作者:管宁  

//站点:www.cndev-lab.com  

//所有稿件均有版权,如要转载,请务必著名出处和作者  

  

#include   

using namespace std;  

  

int main()   

{  

    int a;  

    cin>>a;  

    cout<

    if(cin.good())  

    {  

        cout<<"输入数据的类型正确,无错误!"<

    }  

    if(cin.fail())  

    {  

        cout<<"输入数据类型错误,非致命错误,可清除输入缓冲区挽回!"<

    }  

    system("pause");  

}

  如果错误发生,那么流状态既被标记为错误,你必须清除这些错误状态,以使你的程序能正确适当地继续运行。要清除错误状态,需使用clear()函数。此函数带一个参数,它是你将要设为当前状态的标志值。,只要将ios::goodbit作为实参。 

  示例代码如下:

//程序作者:管宁  

//站点:www.cndev-lab.com  

//所有稿件均有版权,如要转载,请务必著名出处和作者  

  

#include   

using namespace std;  

  

int main()   

{  

    int a;  

    cin>>a;  

    cout<

    cin.clear(ios::goodbit);  

    cout<

    system("pause");  

}

  通常当我们发现输入有错又需要改正的时候,使用clear()更改标记为正确后,同时也需要使用get()成员函数清除输入缓冲区,以达到重复输入的目的。 

  示例代码如下:

//程序作者:管宁  

//站点:www.cndev-lab.com  

//所有稿件均有版权,如要转载,请务必著名出处和作者  

#include   

using namespace std;  

  

int main()   

{  

    int a;  

    while(1)  

    {  

        cin>>a;  

        if(!cin)//条件可改写为cin.fail() 

        {  

            cout<<"输入有错!请重新输入"<

            cin.clear();  

            cin.get();  

        }  

        else  

        {  

            cout<

            break;  

        }  

    }  

    system("pause");  

}

  最后再给出一个对文件流错误标记处理的例子,巩固学习,代码如下:

//程序作者:管宁  

//站点:www.cndev-lab.com  

//所有稿件均有版权,如要转载,请务必著名出处和作者  

#include   

#include   

using namespace std;  

  

int main()   

{  

    ifstream myfile("c:\\1.txt",ios_base::in,0);  

    if(myfile.fail())  

    {  

        cout<<"文件读取失败或指定文件不存在!"<

    }  

    else  

    {  

        char ch;  

        while(myfile.get(ch))  

        {  

            cout<

        }  

        if(myfile.eof())  

        {  

            cout<<"文件内容已经全部读完"<

        }  

        while(myfile.get(ch))  

        {  

            cout<

        }  

    }  

    system("pause");  

}

c++ 中 stringstream介绍

2010-01-28 01:03

C++标准库中的提供了比ANSI C的更高级的一些功能,即单纯性、类型安全和可扩展性。在本文中,我将展示怎样使用这些库来实现安全和自动的类型转换。 

为什么要学习

如果你已习惯了风格的转换,也许你首先会问:为什么要花额外的精力来学习基于的类型 转换呢?也许对下面一个简单的例子的回顾能够说服你。假设你想用sprintf()函数将一个变量从int类型转换到字符串类型。为了正确地完成这个任 务,你必须确保证目标缓冲区有足够大空间以容纳转换完的字符串。此外,还必须使用正确的格式化符。如果使用了不正确的格式化符,会导致非预知的后果。下面 是一个例子:

int n=10000;

chars[10];

sprintf(s,”%d”,n);// s中的内容为“10000”

到目前为止看起来还不错。但是,对上面代码的一个微小的改变就会使程序崩溃:

int n=10000;

char s[10];

sprintf(s,”%f”,n);// 看!错误的格式化符

在这种情况下,程序员错误地使用了%f格式化符来替代了%d。因此,s在调用完sprintf()后包含了一个不确定的字符串。要是能自动推导出正确的类型,那不是更好吗?

进入stringstream

由于n和s的类型在编译期 就确定了,所以编译器拥有足够的信息来判断需要哪些转换。库中声明的标准类就利用了这一点,自动选择所必需的转换。而且, 转换结果保存在stringstream对象的内部缓冲中。你不必担心缓冲区溢出,因为这些对象会根据需要自动分配存储空间。

你的编译器支持吗?

 库是最近才被列入C++标准的。(不要把与标准发布前被删掉的弄混了。)因此,老一点 的编译器,如GCC2.95,并不支持它。如果你恰好正在使用这样的编译器而又想使用的话,就要先对它进行升级更新。

 库定义了三种类:istringstream、ostringstream和stringstream,分别用来进行流的输入、输出和输入输出操作。另 外,每个类都有一个对应的宽字符集版本。简单起见,我主要以stringstream为中心,因为每个转换都要涉及到输入和输出操作。

注意,使用string对象来代替字符数组。这样可以避免缓冲区溢出的危险。而且,传入参数和目标对象的类型被自动推导出来,即使使用了不正确的格式化符也没有危险。

string到int的转换

string result=”10000”;
int n=0;
stream< stream>>n;//n等于10000

重复利用stringstream对象

如果你打算在多次转换中使用同一个stringstream对象,记住再每次转换前要使用clear()方法;

在多次转换中重复使用同一个stringstream(而不是每次都创建一个新的对象)对象最大的好处在于效率。stringstream对象的构造和析构函数通常是非常耗费CPU时间的。

在类型转换中使用模板

你可以轻松地定义函数模板来将一个任意的类型转换到特定的目标类型。例如,需要将各种数字值,如int、long、double等等转换成字符串,要使用以一个string类型和一个任意值t为参数的to_string()函数。to_string()函数将t转换为字符串并写入result中。使用str()成员函数来获取流内部缓冲的一份拷贝:

template

void to_string(string & result,const T& t)

{

ostringstream oss;//创建一个流

oss<

result=oss.str();//获取转换后的字符转并将其写入result
}

这样,你就可以轻松地将多种数值转换成字符串了:

to_string(s1,10.5);//double到string

to_string(s2,123);//int到string

to_string(s3,true);//bool到string

可以更进一步定义一个通用的转换模板,用于任意类型之间的转换。函数模板convert()含有两个模板参数out_type和in_value,功能是将in_value值转换成out_type类型:

template

out_type convert(const in_value & t)

{

stringstream stream;

stream<

out_type result;//这里存储转换结果

stream>>result;//向result中写入值

return result;

}

这样使用convert():

double d;

string salary;

string s=”12.56”;

d=convert(s);//d等于12.56

salary=convert(9000.0);//salary等于”9000”

结论

在过去留下来的程序代码和纯粹的C程序中,传统的形式的转换伴随了我们很长的一段时间。但是,如文中所述,基于 stringstream的转换拥有类型安全和不会溢出这样抢眼的特性,使我们有充足得理由抛弃而使 用库还提供了另外一个特性—可扩展性。你可以通过重载来支持自定义类型间的转换。

一些实例:

stringstream通常是用来做数据转换的。

相比c库的转换,它更加安全,自动和直接。

例子一:基本数据类型转换例子 int转string

#include 
#include 
#include  

int main()
{
std::stringstream stream;
std::string result;
int i = 1000;
stream << i; //将int输入流
     stream >> result; //从stream中抽取前面插入的int值
     std::cout << result << std::endl; // print the string "1000"

运行结果:

例子二:除了基本类型的转换,也支持char *的转换。

#include 
#include  

int main()
{
std::stringstream stream;
char result[8] ;
stream << 8888; //向stream中插入8888
     stream >> result; //抽取stream中的值到result
     std::cout << result << std::endl; // 屏幕显示 "8888"

例子三:再进行多次转换的时候,必须调用stringstream的成员函数clear().

#include 
#include 
int main()
{
std::stringstream stream;
int first, second;
stream<< "456"; //插入字符串
     stream >> first; //转换成int
     std::cout << first << std::endl;
stream.clear(); //在进行多次转换前,必须清除stream
     stream << true; //插入bool值
     stream >> second; //提取出int
     std::cout << second << std::endl;

运行clear的结果

没有运行clear的结果

你可能感兴趣的:(c++笔记)