【本文來自】http://www.builder.com.cn/2003/0304/83250.shtml
http://www.shnenglu.com/alantop/archive/2007/07/10/27823.html
使用stringstream對(duì)象簡(jiǎn)化類型轉(zhuǎn)換
C++標(biāo)準(zhǔn)庫中的<sstream>提供了比ANSI C的<stdio.h>更高級(jí)的一些功能,即單純性、類型安全和可擴(kuò)展性。在本文中,我將展示怎樣使用這些庫來實(shí)現(xiàn)安全和自動(dòng)的類型轉(zhuǎn)換。
為什么要學(xué)習(xí)
如果你已習(xí)慣了<stdio.h>風(fēng)格的轉(zhuǎn)換,也許你首先會(huì)問:為什么要花額外的精力來學(xué)習(xí)基于<sstream>的類型轉(zhuǎn)換呢?也許對(duì)下面一個(gè)簡(jiǎn)單的例子的回顧能夠說服你。假設(shè)你想用sprintf()函數(shù)將一個(gè)變量從int類型轉(zhuǎn)換到字符串類型。為了正確地完成這個(gè)任務(wù),你必須確保證目標(biāo)緩沖區(qū)有足夠大空間以容納轉(zhuǎn)換完的字符串。此外,還必須使用正確的格式化符。如果使用了不正確的格式化符,會(huì)導(dǎo)致非預(yù)知的后果。下面是一個(gè)例子:
int n=10000;
chars[10];
sprintf(s,”%d”,n);// s中的內(nèi)容為“10000”
到目前為止看起來還不錯(cuò)。但是,對(duì)上面代碼的一個(gè)微小的改變就會(huì)使程序崩潰:
int n=10000;
char s[10];
sprintf(s,”%f”,n);// 看!錯(cuò)誤的格式化符
在這種情況下,程序員錯(cuò)誤地使用了%f格式化符來替代了%d。因此,s在調(diào)用完sprintf()后包含了一個(gè)不確定的字符串。要是能自動(dòng)推導(dǎo)出正確的類型,那不是更好嗎?
進(jìn)入stringstream
由于n和s的類型在編譯期就確定了,所以編譯器擁有足夠的信息來判斷需要哪些轉(zhuǎn)換。<sstream>庫中聲明的標(biāo)準(zhǔn)類就利用了這一點(diǎn),自動(dòng)選擇所必需的轉(zhuǎn)換。而且,轉(zhuǎn)換結(jié)果保存在stringstream對(duì)象的內(nèi)部緩沖中。你不必?fù)?dān)心緩沖區(qū)溢出,因?yàn)檫@些對(duì)象會(huì)根據(jù)需要自動(dòng)分配存儲(chǔ)空間。
你的編譯器支持<sstream>嗎?
<sstream>庫是最近才被列入C++標(biāo)準(zhǔn)的。(不要把<sstream>與標(biāo)準(zhǔn)發(fā)布前被刪掉的<strstream>弄混了。)因此,老一點(diǎn)的編譯器,如GCC2.95,并不支持它。如果你恰好正在使用這樣的編譯器而又想使用<sstream>的話,就要先對(duì)它進(jìn)行升級(jí)更新。
<sstream>庫定義了三種類:istringstream、ostringstream和stringstream,分別用來進(jìn)行流的輸入、輸出和輸入輸出操作。另外,每個(gè)類都有一個(gè)對(duì)應(yīng)的寬字符集版本。簡(jiǎn)單起見,我主要以stringstream為中心,因?yàn)槊總€(gè)轉(zhuǎn)換都要涉及到輸入和輸出操作。
注意,<sstream>使用string對(duì)象來代替字符數(shù)組。這樣可以避免緩沖區(qū)溢出的危險(xiǎn)。而且,傳入?yún)?shù)和目標(biāo)對(duì)象的類型被自動(dòng)推導(dǎo)出來,即使使用了不正確的格式化符也沒有危險(xiǎn)。
string到int的轉(zhuǎn)換
string result=”10000”;
int n=0;
stream<<result;
stream>>n;//n等于10000
重復(fù)利用stringstream對(duì)象
如果你打算在多次轉(zhuǎn)換中使用同一個(gè)stringstream對(duì)象,記住再每次轉(zhuǎn)換前要使用clear()方法;
在多次轉(zhuǎn)換中重復(fù)使用同一個(gè)stringstream(而不是每次都創(chuàng)建一個(gè)新的對(duì)象)對(duì)象最大的好處在于效率。stringstream對(duì)象的構(gòu)造和析構(gòu)函數(shù)通常是非常耗費(fèi)CPU時(shí)間的。
在類型轉(zhuǎn)換中使用模板
你可以輕松地定義函數(shù)模板來將一個(gè)任意的類型轉(zhuǎn)換到特定的目標(biāo)類型。例如,需要將各種數(shù)字值,如int、long、double等等轉(zhuǎn)換成字符串,要使用以一個(gè)string類型和一個(gè)任意值t為參數(shù)的to_string()函數(shù)。to_string()函數(shù)將t轉(zhuǎn)換為字符串并寫入result中。使用str()成員函數(shù)來獲取流內(nèi)部緩沖的一份拷貝:
template<class T>
void to_string(string & result,const T& t)
{
ostringstream oss;//創(chuàng)建一個(gè)流
oss<<t;//把值傳遞如流中
result=oss.str();//獲取轉(zhuǎn)換后的字符轉(zhuǎn)并將其寫入result
}
這樣,你就可以輕松地將多種數(shù)值轉(zhuǎn)換成字符串了:
to_string(s1,10.5);//double到string
to_string(s2,123);//int到string
to_string(s3,true);//bool到string
可以更進(jìn)一步定義一個(gè)通用的轉(zhuǎn)換模板,用于任意類型之間的轉(zhuǎn)換。函數(shù)模板convert()含有兩個(gè)模板參數(shù)out_type和in_value,功能是將in_value值轉(zhuǎn)換成out_type類型:
template<class out_type,class in_value>
out_type convert(const in_value & t)
{
stringstream stream;
stream<<t;//向流中傳值
out_type result;//這里存儲(chǔ)轉(zhuǎn)換結(jié)果
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”
結(jié)論
在過去留下來的程序代碼和純粹的C程序中,傳統(tǒng)的<stdio.h>形式的轉(zhuǎn)換伴隨了我們很長(zhǎng)的一段時(shí)間。但是,如文中所述,基于stringstream的轉(zhuǎn)換擁有類型安全和不會(huì)溢出這樣搶眼的特性,使我們有充足得理由拋棄<stdio.h>而使用<sstream>。<sstream>庫還提供了另外一個(gè)特性—可擴(kuò)展性。你可以通過重載來支持自定義類型間的轉(zhuǎn)換。
一些實(shí)例:
stringstream通常是用來做數(shù)據(jù)轉(zhuǎn)換的。
相比c庫的轉(zhuǎn)換,它更加安全,自動(dòng)和直接。
例子一:基本數(shù)據(jù)類型轉(zhuǎn)換例子 int轉(zhuǎn)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"
}
運(yùn)行結(jié)果:
例子二:除了基本類型的轉(zhuǎn)換,也支持char *的轉(zhuǎn)換。
#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"
}
例子三:再進(jìn)行多次轉(zhuǎn)換的時(shí)候,必須調(diào)用stringstream的成員函數(shù)clear().
#include <sstream>
#include <iostream>
int main()
{
std::stringstream stream;
int first, second;
stream<< "456"; //插入字符串
stream >> first; //轉(zhuǎn)換成int
std::cout << first << std::endl;
stream.clear(); //在進(jìn)行多次轉(zhuǎn)換前,必須清除stream
stream << true; //插入bool值
stream >> second; //提取出int
std::cout << second << std::endl;
}
運(yùn)行clear的結(jié)果
沒有運(yùn)行clear的結(jié)果
