使用stringstream對象簡化類型轉換
C++標準庫中的<sstream>提供了比ANSI C的<stdio.h>更高級的一些功能,即單純性、類型安全和可擴展性。在本文中,我將展現怎樣使用這些庫來實現安全和自動的類型轉換。ios
爲何要學習程序員
若是你已習慣了<stdio.h>風格的轉換,也許你首先會問:爲何要花額外的精力來學習基於<sstream>的類型轉換呢?也許對下面一個簡單的例子的回顧可以說服你。假設你想用sprintf()函數將一個變量從int類型轉換到字符串類型。爲了正確地完成這個任務,你必須確保證目標緩衝區有足夠大空間以容納轉換完的字符串。此外,還必須使用正確的格式化符。若是使用了不正確的格式化符,會致使非預知的後果。下面是一個例子:數組
int n=10000;安全
chars[10];函數
sprintf(s,」%d」,n);// s中的內容爲「10000」
到目前爲止看起來還不錯。可是,對上面代碼的一個微小的改變就會使程序崩潰:
int n=10000;學習
char s[10];3d
sprintf(s,」%f」,n);// 看!錯誤的格式化符對象
在這種狀況下,程序員錯誤地使用了%f格式化符來替代了%d。所以,s在調用完sprintf()後包含了一個不肯定的字符串。要是能自動推導出正確的類型,那不是更好嗎?blog
進入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
運行結果:
例子二:除了基本類型的轉換,也支持char *的轉換。
例子三:再進行屢次轉換的時候,必須調用stringstream的成員函數clear().
運行clear的結果
沒有運行clear的結果