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)


相关推荐

  • 第二章,springboot 配置,yaml 语法[通俗易懂]

    第二章,springboot 配置,yaml 语法[通俗易懂]第二章,springboot 配置,yaml 语法

  • datagrip2021.7.15 最新激活码(注册激活)

    (datagrip2021.7.15 最新激活码)2021最新分享一个能用的的激活码出来,希望能帮到需要激活的朋友。目前这个是能用的,但是用的人多了之后也会失效,会不定时更新的,大家持续关注此网站~https://javaforall.cn/100143.htmlIntelliJ2021最新激活注册码,破解教程可免费永久激活,亲测有效,上面是详细链接哦~ML…

  • fbx文件导入3dmax_3d中z轴的值没办法输入

    fbx文件导入3dmax_3d中z轴的值没办法输入本文通过参考网上资源做的一个例子。本程序的功能就是通过xna将3d图像显示到winfrom对他进行旋转操作。首先我们先准备好两个文件夹model文件夹放fbx文件,textures放渲染文件,操作步骤都是添加现有项,准备好资源文件后,先检查下是否有以下引用下面将定义Ga…

  • 12306自动刷票下单-登录篇(一)

    12306自动刷票下单-登录篇(一)12306网站推出图片验证码以后,对于抢票软件就提出了更高的要求,本篇并不涉及自动识别验证码登录(主要是博主能力所限),提供一个途径-打码平台,这个几乎是可以激活成功教程所有验证码了,本篇主要是分享一下12306网站登录的流程的学习,勿吐槽,有问题请指正,博主也是刚开始接触爬虫,大家共勉共勉。废话不多说了,直接干吧 首先打开12306登录页面https://kyfw.12306.cn/otn/login/…

    2022年10月23日
  • 共射极放大电路和共基极放大电路_如何判断放大电路是共集还是共射

    共射极放大电路和共基极放大电路_如何判断放大电路是共集还是共射如何区分共射极放大电路与共基极放大电路?_百度知道如何区分共射极放大电路与共基极放大电路?_百度知道答有简单的方法:观察信号的输入端和输出端,就看信号正极。共射电路:信号从基极进入,从集电极

  • 【软件工程】——详细设计说明书「建议收藏」

    【软件工程】——详细设计说明书「建议收藏」1引言1.1编写目的该文档在概要设计的基础上,进一步的细化系统结构,展示了软件结构的图标,物理设计、数据结构设计、及算法设计、详细的介绍了系统各个模块是如何实现的,包括涉及到的算法,逻辑流程等。预期的读者:程序员1.2背景a. 待开发软件系统的名称:机房收费系统b. 项目的任务提出者:米新江教授c. 项目的开发者:齐智d. 项目的用户:廊坊师范学院全体在职员工及学生e. 运行该软…

发表回复

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

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