天天看點

C++ 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庫的轉換,它更加安全,自動和直接。

 注意:stringstream 清空操作

stringstream str;//char *

str.clear();

str.str("");

繼續閱讀