stringstream 的用法介绍[通俗易懂]

stringstream 的用法介绍[通俗易懂]stringstream主要有两个作用:简化类型转换和一次性读入数据: 一、使用stringstream对象简化类型转换C++标准库中的提供了比ANSIC的更高级的一些功能,即单纯性、类型安全和可扩展性。在本文中,我将展示怎样使用这些库来实现安全和自动的类型转换。为什么要学习如果你已习惯了风格的转换,也许你首先会问:为什么要花额外的精力来学习基于的类型转换呢?也许对下面一

大家好,又见面了,我是你们的朋友全栈君。

stringstream 主要有两个作用:简化类型转换和一次性读入数据:

 

一、使用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

由于n 和s 的类型在编译期 就确定了,所以编译器拥有足够的信息来判断需要哪些转换。<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; //print 456
    stream
 . clear();  // 在进行多次转换前,必须清除stream 
    stream  <<   true  // 插入bool值 
    stream  >>  second;  // 提取出int 
    std :: cout  <<  second  <<  std :: endl;  // print 1

 


二、一次性将文件全部读入内存的方法之一:

 

       很多时候较大数据 量的文件 IO 总是成为瓶颈,为了提高效率,有时想要先将文件大块大块的读入再行处理。下面分析两种惯常的处理手法。

1. 将文件一次性读入 string 中。

        貌似 std::getline 、 istream::getline 或是 operator<< operator>> 等都不提供一次读到文件结尾的机制,只有 istreambuf_iterator 可以做到:

{

ifstream in( “input.txt” ) ;
string instring( ( istreambuf_iterator<char>( in) ) , istreambuf_iterator<char>( ) ) ;

}

    

string 的构造函数 前一个参数要多加一层 () 以免编译器误认为是函数声明 = = …

这样读入 string 会随着内容动态增长,空间不足时会触发额外的 realloc 及 copy 操作,为提高效率有必要预分配足够的空间:

{

    ifstream in( “input.txt” ) ;

    in.seekg ( 0 , ios::end ) ;
    streampos len = in.tellg ( ) ;
    in.seekg ( 0 , ios::beg ) ;

    string instring;
    instring.reserve ( len) ;
    instring.assign ( istreambuf_iterator<char>( in) , istreambuf_iterator<char>( ) ) ;

}


2. 将文件一次性读入 stringstream 中。

        filebuf 和 stringbuf 无法直接通过 rdbuf() 重定向,因此从 filebuf 到 stringbuf 需要一次 copy 操作。最简单的方法是直接复制整个 streambuf :

{

    ifstream in( “input.txt” ) ;
    stringstream ss;
    ss<<in.rdbuf ( ) ;

}


与 string 的情况相同,这里同样也有一个空间 realloc 及 copy 的问题。但 streambuf 的缓冲区不是那么方便操作的,解决方法是我们给他手动指定一个空间:

{

    ifstream in( “input.txt” ) ;
    in.seekg ( 0 , ios::end ) ;
    streampos len = in.tellg ( ) ;
    in.seekg ( 0 , ios::beg ) ;
    vector<char> buffer( len) ;
    in.read ( &buffer[ 0 ] , len) ;

    stringstream ss;
    ss.rdbuf ( ) ->pubsetbuf( &buffer[ 0 ] , len) ;

}


例如:

文件test内容为:

A    0.1
B    1
C    0.33


一次性读入该文件:


#include <iostream>

#include <fstream>

#include <string>

#include <sstream>

int main()

{

    ifstream in(“test”);

   

    in.seekg(0,ios::end);
    streampos len=in.tellg();
    vector<char> buffer(len);
    in.seekg(0,ios::beg);
    in.read(&buffer[0],len);

   

    stringstream ss;

    ss.rdbuf()->pubsetbuf(&buffer[0],len);

    string name;
    double rate;

    while (ss >> name && ss >> rate)
    {

        cout << name <<“/t” << rate << std::endl;
    }
    return 0;
}

 

输出为:

A    0.1
B    1
C    0.33

 


 From: http://blog.csdn.net/computer055maxi/article/details/6218873

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

发布者:全栈程序员-用户IM,转载请注明出处:https://javaforall.cn/138922.html原文链接:https://javaforall.cn

【正版授权,激活自己账号】: Jetbrains全家桶Ide使用,1年售后保障,每天仅需1毛

【官方授权 正版激活】: 官方授权 正版激活 支持Jetbrains家族下所有IDE 使用个人JB账号...

(0)


相关推荐

  • Java xml格式化工具「建议收藏」

    Java xml格式化工具「建议收藏」Javaxml格式化工具Java实现xml格式化工具代码地址:https://github.com/xiaxveliang/JavaTool_XmlValidate运行效果点击“乘1.5”按钮后的运行效果

  • 怎样在SharePoint管理中心检查数据库架构版本号、修补级别和修补程序的常规监控

    怎样在SharePoint管理中心检查数据库架构版本号、修补级别和修补程序的常规监控

  • 错误:备份集中的数据库备份与现有XXX数据库不同

    错误:备份集中的数据库备份与现有XXX数据库不同当在使用另外一台的数据库备份文件.bak恢复到本机数据库时,遇到“备份集中的数据库备份与现有XXX数据库不同”的错误时,只需要以下几步:删除与要恢复数据库同名的已经存在的数据库; 右击“数据库”选择“还原数据库”;选择源设备;然后再在弹出窗口的“目标数据库”中选择要还原数据库名;这时要备份的数据库的名字会弹出来,所以不需要提前创建现在手中有SqlServer2…

  • 凯福德金业(香港)荣获“最佳贵金属交易平台”大奖

    凯福德金业(香港)荣获“最佳贵金属交易平台”大奖凯福德金业(香港)荣获“最佳贵金属交易平台”大奖 “2013年度第二届资本力量百强榜评选”中,凯福德金业凭借其在推动贵金属行业发展中作出的努力与贡献,成功入选“2013资本力量百强”并被授予“最佳贵金属交易平台”大奖。2014年2月,证券之星2013年第二届“资本力量百强榜颁布盛典”在上海如期举行。此次评选由中国著名财经资讯网站证券之星主办,联合复旦大学管理学院、证监会、上市公司联合会、逾

  • 大数据分析技术方案

    大数据分析技术方案现在已经进入大数据时代,数据是无缝连接网络世界与物理世界的DNA。发现数据DNA、重组数据DNA是人类不断认识、探索、实践大数据的持续过程。大数据分析可以有效地促进营销,个性化医疗治病,帮助学生提高

  • c语言中char类型转int类型「建议收藏」

    c语言中char类型转int类型「建议收藏」前言在九度oj做acm的时候,经常会遇到了char类型和int类型相互转化的问题,这里进行一下总结。今后,可能会多次更新博客,因为半年做了很多总结,但是都是保存在word文档上了,现在开始慢慢向CSDN博客转移。问题类型char型数字转换为int型转换方法a[i]-‘0’参考程序#include#include#includei

发表回复

您的电子邮箱地址不会被公开。

关注全栈程序员社区公众号