istringstream、ostringstream、stringstream 类介绍 和 stringstream类 clear函数的真正用途
来源: http://blog.csdn.net/TQH_Candy/article/details/52494570
0、C++的输入输出分为三种:
(1)基于控制台的I/O
(2)基于文件的I/O
(3)基于字符串的I/O
1、头文件
- #include <sstream>
2、作用
istringstream类用于执行C++风格的字符串流的输入操作。
ostringstream类用于执行C++风格的字符串流的输出操作。
strstream类同时可以支持C++风格的串流的输入输出操作。
3、具体分析
istringstream类
描述:从流中提取数据,支持 >> 操作
这里字符串可以包括多个单词,单词之间使用空格分开
- istringstream的构造函数原形:
- istringstream::istringstream(string str);
初始化:使用字符串进行初始化
- istringstream istr("1 56.7");
- istr.str("1 56.7");//把字符串"1 56.7"存入字符串流中
使用:我们可以使用分解点获取不同的数据,完成 字符串 到 其他类型 的转换
常用成员函数:
- str():使istringstream对象返回一个string字符串
举例:把字符串类型的数据转换为其他类型
- #include <iostream>
- #include <sstream>
- using namespace std;
- int main()
- {
- istringstream istr("1 56.7");
- cout<<istr.str()<<endl;//直接输出字符串的数据 "1 56.7"
- string str = istr.str();//函数str()返回一个字符串
- cout<<str<<endl;
- int n;
- double d;
- //以空格为界,把istringstream中数据取出,应进行类型转换
- istr>>n;//第一个数为整型数据,输出1
- istr>>d;//第二个数位浮点数,输出56.7
- //假设换下存储类型
- istr>>d;//istringstream第一个数要自动变成浮点型,输出仍为1
- istr>>n;//istringstream第二个数要自动变成整型,有数字的阶段,输出为56
- //测试输出
- cout<<d<<endl;
- cout<<n<<endl;
- system("pause");
- return 1;
- }
举例2:把一行字符串放入流中,单词以空格隔开。之后把一个个单词从流中依次读取到字符串
- #include <iostream>
- #include <sstream>
- using namespace std;
- int main()
- {
- istringstream istr;
- string line,str;
- while (getline(cin,line))//从终端接收一行字符串,并放入字符串line中
- {
- istr.str(line);//把line中的字符串存入字符串流中
- while(istr >> str)//每次读取一个单词(以空格为界),存入str中
- {
- cout<<str<<endl;
- }
- }
- system("pause");
- return 1;
- }
输入:123 34 45
输出:
123 换行 34 换行 45
ostringstream类
描述:把其他类型的数据写入流(往流中写入数据),支持<<操作
- ostringstream的构造函数原形:
- ostringstream::ostringstream(string str);
初始化:使用字符串进行初始化
- ostringstream ostr("1234");
- ostr.str("1234");//把字符串"1234"存入字符串流中
举例:
- #include <iostream>
- #include <sstream>
- using namespace std;
- int main()
- {
- //初始化输出字符串流ostr
- ostringstream ostr("1234");
- cout<<ostr.str()<<endl;//输出1234
- ostr.put(‘5‘);//字符4顶替了1的位置
- cout<<ostr.str()<<endl;//输出5234
- ostr<<"67";//字符串67替代了23的位置,输出5674
- string str = ostr.str();
- cout<<str<<endl;
- system("pause");
- return 1;
- }
stringstream类
描述:是对istringstream和ostringstream类的综合,支持<<, >>操作符,可以进行字符串到其它类型的快速转换
- stringstream的构造函数原形如下:
- stringstream::stringstream(string str);
初始化:使用字符串进行初始化
- stringstream str("1234");
- str.str("1234");//把字符串"1234"存入字符串流中
作用:
1、stringstream通常是用来做数据转换的
2、将文件的所有数据一次性读入内存
举例1:基本数据类型变字符串
- /*基本数据类型变字符串*/
- #include <fstream>
- #include <iostream>
- #include <sstream>
- using namespace std;
- int main()
- {
- /*整型变字符串*/
- int n = 10;
- string str;
- stringstream stream;
- stream << n;
- stream >> str;
- cout<<str<<endl;
- stream.clear();//多次使用stringstream,要先清空下,不能使用stream.str("");否则下面输出10
- /*char* 变 string*/
- char cStr[10] = "china";
- stream << cStr;
- stream >> str;
- cout<<str<<endl;
- system("pause");
- return 1;
- }
举例2:字符串变基本数据类型
- /*字符串变基本数据类型*/
- #include <fstream>
- #include <iostream>
- #include <sstream>
- using namespace std;
- int main()
- {
- /*字符串 变 double*/
- double n;
- string str = "12.5";
- stringstream stream;
- stream << str;
- stream >> n;
- cout<<n<<endl;
- stream.clear();//多次使用stringstream,要先清空下,不能使用stream.str("");
- /*string 变 char* */
- string str1 = "china";
- char cStr[10];
- stream << str1;
- stream >> cStr;
- cout<<cStr<<endl;//输出china
- system("pause");
- return 1;
- }
- 注意:
- #include <iostream>
- #include <sstream>
- using namespace std;
- int main(int argc,char *argv[])
- {
- std::stringstream stream;
- string str;
- while(1)
- {
- //clear(),这个名字让很多人想当然地认为它会清除流的内容。
- //实际上,它并不清空任何内容,它只是重置了流的状态标志而已!
- stream.clear();
- // 去掉下面这行注释,清空stringstream的缓冲,每次循环内存消耗将不再增加!
- //stream.str("");
- stream<<"sdfsdfdsfsadfsdafsdfsdgsdgsdgsadgdsgsdagasdgsdagsadgsdgsgdsagsadgs";
- stream>>str;
- //测试输出每次循环,你的内存消耗增加了多少!
- cout<<"Size of stream = "<<stream.str().length()<<endl;
- system("PAUSE");
- }
- system("PAUSE");
- return EXIT_SUCCESS;
- }
由于stringstream构造函数会特别消耗内存,似乎不打算主动释放内存(或许是为了提高效率),但如果你要在程序中用同一个流,反复读写大量的数据,将会造成大量的内存消耗,因些这时候,需要适时地清除一下缓冲 (用 stream.str("") )。
另外不要企图用 stream.str().resize(0),或 stream.str().clear() 来清除缓冲,使用它们似乎可以让stringstream的内存消耗不要增长得那么快,但仍然不能达到清除stringstream缓冲的效果,内存的消耗还在缓慢的增长!,至于stream.flush(),则根本就起不到任何作用。
//=============================================
究竟什么情况下需要用到clear
先来看一个stack overflow上的问题(http://stackoverflow.com/questions/35080342/using-two-string-streams-but-not-getting-same-result)
我将其简化为以下代码:
- int main() {
- string line = "1 2 3 4 5";
- stringstream s1(line);
- string temp;
- int toAdd;
- stringstream s2;
- while (s1 >> temp) {
- cout << "temp:" << temp << endl;
- s2 << temp;
- cout << "s2.str: " << s2.str() << endl;
- s2 >> toAdd;
- cout << "toAdd:" << toAdd << endl;
- s2.str("");
- }
- return 0;
- }
这个代码的原意是要把line中的字符串形式的1 2 3 4 5一个一个地转成int并输出,所以我们期望的toAdd的输出应该是1 2 3 4 5,但结果却是 1 1 1 1 1, 如下图
可以从s2.str:这句输出中看到, 只有第一次是正常地把temp输入进s2,后面的都失败了。
原因在于, s2在第一次调用完operator<<和operator>>后,来到了end-of-file的位置,此时stringstream会为其设置一个eofbit的标记位,标记其为已经到达eof。查文档得知, 当stringstream设置了eofbit,任何读取eof的操作都会失败,同时,会设置failbit的标记位,标记为失败状态。所以后面的操作都失败了,toAdd的值一直都是1。
Operations that attempt to read at the End-of-File fail, and thus both the eofbit and the failbit end up set. This function can be used to check whether the failure is due to reaching the End-of-File or to some other reason.
clear函数:
原型: void clear (iostate state = goodbit);
标志位一共有4种, goodbit, eofbit, failbit, badbit
clear可以清除掉所有的error state
- int main() {
- string line = "1 2 3 4 5";
- stringstream s1(line);
- string temp;
- int toAdd;
- stringstream s2;
- while (s1 >> temp) {
- cout << "temp:" << temp << endl;
- s2 << temp;
- cout << "s2.str: " << s2.str() << endl;
- s2 >> toAdd;
- cout << "toAdd:" << toAdd << endl;
- s2.str("");
- if (s2.eof()) {
- s2.clear();
- cout << "s2.eof true" << endl;
- }
- }
- return 0;
- }
使用clear后, s2就可以正常地工作了,结果如下:
参考网站:http://www.cplusplus.com/reference/sstream/stringstream/
http://www.cplusplus.com/reference/ios/ios/clear/