关闭

istringstream,ostringstream,stringstream<sstream>

78人阅读 评论(0) 收藏 举报


转自:http://www.cnblogs.com/likebeta/archive/2012/07/24/2607397.html


字符串流 istringstream 和 ostringstream 的用法


iostream 标准库支持内存中的输入/输出,只要将流与存储在程序内存中的 string 对象捆绑起来即可。此时,可使用 iostream 输入和输出操作符读写这个 string 对象。标准库定义了三种类型的字符串流:
•   istringstream,由 istream 派生而来,提供读 string 的功能。
•   ostringstream,由 ostream 派生而来,提供写 string 的功能。
•   stringstream,由 iostream 派生而来,提供读写 string 的功能。
要使用上述类,必须包含 sstream 头文件。
与 fstream 类型一样,上述类型由 iostream 类型派生而来,这意味着 iostream 上所有的操作适用于 sstream 中的类型。sstream 类型除了继承的操作外,还各自定义了一个有 string 形参的构造函数,这个构造函数将 string 类型的实参复制给 stringstream 对象。对 stringstream 的读写操作实际上读写的就是该对象中的 string 对象。这些类还定义了名为 str 的成员,用来读取或设置 stringstream 对象所操纵的 string 值。

注意到尽管 fstream 和 sstream 共享相同的基类,但它们没有其他相互关
系。特别是,stringstream 对象不使用 open 和 close 函数,而 fstream 对
象则不允许使用 str。
stringstream  特定的操作

stringstream strm; // 创建自由的 stringstream 对象
stringstream strm(s); //创建存储 s 的副本的 stringstream 对象,其中 s 是 string 类型的对象
strm.str()  //返回 strm 中存储的 string 类型对象
strm.str(s)  //将 string 类型的 s 复制给 strm,返回 void 

 stringstream  对象的和使用
前面已经见过以每次一个单词或每次一行的方式处理输入的程序。第一种程
序用 string 输入操作符,而第二种则使用 getline 函数。然而,有些程序需
要同时使用这两种方式:有些处理基于每行实现,而其他处理则要操纵每行中每
个单词。可用 stringstreams 对象实现:

复制代码
string line, word;      // will hold a line and word from input,respectively
while (getline(cin, line))
{
    // read a line from theinput into line do per-line processing
    istringstream stream(line);  // bind to stream to the
    line we read
    while (stream >> word)
    { 
         // read a word from line
         // do per-word processing
    }
}
复制代码

这里,使用 getline 函数从输入读取整行内容。然后为了获得每行中的单词,将一个 istringstream 对象与所读取的行绑定起来,这样只需要使用普通的 string 输入操作符即可读出每行中的单词。

stringstream  提供的转换和/或格式化
stringstream 对象的一个常见用法是,需要在多种数据类型之间实现自动格式化时使用该类类型。例如,有一个数值型数据集合,要获取它们的 string 表示形式,或反之。sstream 输入和输出操作可自动地把算术类型转化为相应的 string 表示形式,反过来也可以。

int val1 = 512, val2 = 1024;
ostringstream format_message;
// ok: converts values to a string representation
format_message << "val1: " << val1 << "\n"<< "val2: " << val2 << "\n"; 

这里创建了一个名为 format_message 的 ostringstream 类型空对象,并将指定的内容插入该对象。重点在于 int 型值自动转换为等价的可打印的字符串。format_message 的内容是以下字符:

val1: 512\nval2: 1024 

相反,用 istringstream 读 string 对象,即可重新将数值型数据找回来。读取 istringstream 对象自动地将数值型数据的字符表示方式转换为相应的算术值。

// str member obtains the string associated with a stringstream
istringstream input_istring(format_message.str());
string dump; // place to dump the labels from the formatted message
// extracts the stored ascii values, converting back to arithmetic types
input_istring >> dump >> val1 >> dump >> val2;
cout << val1 << " " << val2 << endl;  // prints 512 1024 

这里使用 。str 成员获取与之前创建的 ostringstream 对象关联的 string 副本。再将 input_istring 与 string 绑定起来。在读 input_istring 时,相应的值恢复为它们原来的数值型表示形式为了读取 input_string,必须把该 string 对象分解为若干个部分。我们要的是数值型数据;为了得到它们,必须读取(和忽略)处于所需数据周围的标号。

因为输入操作符读取的是有类型的值,因此读入的对象类型必须和由 stringstream 读入的值的类型一致。在本例中,input_istring 分成四个部分:
string 类型的值 val1,接着是 512,然后是 string 类型的值 val2,最后是 1024。一般情况下,使用输入操作符读 string 时,空白符将会忽略。于是,在读与 format_message 关联的 string 时,忽略其中的换行符。

更多阅读:

http://www.cplusplus.com/reference/iostream/istringstream/

http://www.cplusplus.com/reference/iostream/ostringstream/

 





转载:http://blog.csdn.net/qq1987924/article/details/7671154

ostringstream的用法

【本文来自】http://www.builder.com.cn/2003/0304/83250.shtml
http://www.cppblog.com/alantop/archive/2007/07/10/27823.html
使用stringstream对象简化类型转换
C++标准库中的<sstream>提供了比ANSI C的<stdio.h>更高级的一些功能,即单纯性、类型安全和可扩展性。在本文中,我将展示怎样使用这些库来实现安全和自动的类型转换。

为什么要学习

如果你已习惯了<stdio.h>风格的转换,也许你首先会问:为什么要花额外的精力来学习基于<sstream>的类型转换呢?也许对下面一个简单的例子的回顾能够说服你。假设你想用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

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

你的编译器支持<sstream>吗?

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

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

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

string到int的转换

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

重复利用stringstream对象

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

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

在类型转换中使用模板

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

template<class T>

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

{

ostringstream oss;//创建一个流

oss<<t;//把值传递如流中

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<class out_type,class in_value>

out_type convert(const in_value & t)

{

stringstream stream;

stream<<t;//向流中传值

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

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

return result;

}

这样使用convert():

double d;

string salary;

string s=”12.56”;

d=convert<double>(s);//d等于12.56

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

结论

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

一些实例:

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

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

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

#include <string>
#include <sstream>
#include <iostream>

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 <sstream>
#include <iostream>

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 <sstream>
#include <iostream>
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;
}

一、简单介绍

ostringstream是C++的一个字符集操作模板类,定义在sstream.h头文件中。ostringstream类通常用于执行C风格的串流的输出操作,格式化字符串,避免申请大量的缓冲区,替代sprintf。

派生关系图:


二、ostringstream的基本使用

ostringstream的构造函数形式:
explicit ostringstream ( openmode which = ios_base::out );
explicit ostringstream ( const string & str, openmode which = ios_base::out );


有时候,我们需要格式化一个字符串,但通常并不知道需要多大的缓冲区。为了保险常常申请大量的缓冲区以防止缓冲区过小造成字符串无法全部存储。这时我们可以考虑使用ostringstream类,该类能够根据内容自动分配内存,并且其对内存的管理也是相当的到位。取得std::ostringstream里的内容可以通过str()和str(string&)成员函数。

三、注意事项

std::ostringstream::str()返回的是临时对象,不能对其直接操作。

例如会有如下误用:

const char *  pBuffer  =  oss.str().c_str();
注意pBuffer指向的内存已被析构!!

四、代码测试

  1. #include <sstream>    
  2. #include <string>    
  3. #include <iostream>    
  4. using namespace std;    
  5.     
  6. void main()    
  7. {    
  8.     ostringstream ostr1; // 构造方式1    
  9.     ostringstream ostr2("abc"); // 构造方式2    
  10.     
  11. /*----------------------------------------------------------------------------  
  12. *** 方法str()将缓冲区的内容复制到一个string对象中,并返回  
  13. ----------------------------------------------------------------------------*/    
  14.     ostr1 << "ostr1 " << 2012 << endl; // 格式化,此处endl也将格式化进ostr1中    
  15.     cout << ostr1.str();     
  16.     
  17. /*----------------------------------------------------------------------------  
  18. *** 建议:在用put()方法时,先查看当前put pointer的值,防止误写  
  19. ----------------------------------------------------------------------------*/    
  20.     long curPos = ostr2.tellp(); //返回当前插入的索引位置(即put pointer的值),从0开始     
  21.     cout << "curPos = " << curPos << endl;    
  22.     
  23.     ostr2.seekp(2); // 手动设置put pointer的值    
  24.     ostr2.put('g');     // 在put pointer的位置上写入'g',并将put pointer指向下一个字符位置    
  25.     cout << ostr2.str() << endl;    
  26.         
  27.     
  28. /*----------------------------------------------------------------------------  
  29. *** 重复使用同一个ostringstream对象时,建议:  
  30. *** 1:调用clear()清除当前错误控制状态,其原型为 void clear (iostate state=goodbit);  
  31. *** 2:调用str("")将缓冲区清零,清除脏数据  
  32. ----------------------------------------------------------------------------*/    
  33.     ostr2.clear();    
  34.     ostr2.str("");    
  35.     
  36.     cout << ostr2.str() << endl;    
  37.     ostr2.str("_def");    
  38.     cout << ostr2.str() << endl;    
  39.     ostr2 << "gggghh";    // 覆盖原有的数据,并自动增加缓冲区    
  40.     cout << ostr2.str() << endl;  
  41.     ostr2.str("");   // 若不加这句则运行时错误,因为_df所用空间小于gggghh,导致读取脏数据  
  42.     ostr2.str("_df");    
  43.     cout << ostr2.str() << endl;  
  44.   
  45.     // 输出随机内存值,危险  
  46.     const char* buf = ostr2.str().c_str();    
  47.     cout << buf << endl;  
  48.   
  49.     // 正确输出_df  
  50.     string ss = ostr2.str();  
  51.     const char *buffer = ss.c_str();  
  52.     cout << buffer << endl;  
  53. }  

运行结果如下:

http://blog.csdn.net/lanxuezaipiao/article/details/16358159



0
0

查看评论
* 以上用户言论只代表其个人观点,不代表CSDN网站的观点或立场
    个人资料
    • 访问:10735次
    • 积分:212
    • 等级:
    • 排名:千里之外
    • 原创:6篇
    • 转载:38篇
    • 译文:0篇
    • 评论:0条
    文章分类