使用stringstream物件簡化型別轉換

2021-05-05 10:11:19 字數 3791 閱讀 9302

stringstream是個好東西,網上有不少文章,討論如何用它實現各種資料型別的轉換(比如把double或int轉換為string型別)。但如果stringstream使用不當,當心記憶體出問題(我就吃過虧^_^)。

試試下面的**,執行程式前開啟任務管理器,過不了幾十秒,所有的記憶體都將被耗盡!

#include

#include

#include

using namespace std;

/

int main(int argc,char * argv)

system("pause ");

return exit_success;

}

stream.str("");那一行的注釋去掉,再執行程式,記憶體就正常了

看來stringstream似乎不打算主動釋放記憶體(或許是為了提高效率),但如果你要在程式中用同乙個流,反覆讀寫大量的資料,將會造成大量的記憶體消耗,因些這時候,需要適時地清除一下緩衝 (用 stream.str("") )。

另外不要企圖用 stream.str().resize(0),或 stream.str().clear() 來清除緩衝,使用它們似乎可以讓stringstream的記憶體消耗不要增長得那麼快,但仍然不能達到清除stringstream緩衝的效果(不信做個 實驗就知道了,記憶體的消耗還在緩慢的增長!),至於stream.flush(),則根本就起不到任何作用。

c++標準庫中的提供了比ansi c的更高階的一些功能,即單純性、型別安全和可擴充套件性。在本文中,我將展示怎樣使用這些庫來實現安全和自動的型別轉換。

為什麼要學習

如果你已習慣了風格的轉換,也許你首先會問:為什麼要花額外的精力來學習基於的型別 轉換呢?也許對下面乙個簡單的例子的回顧能夠說服你。假設你想用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 的型別在編譯期 就確定了,所以編譯器擁有足夠的資訊來判斷需要哪些轉換。庫中宣告的標準類就利用了這一點,自動選擇所必需的轉換。而且, 轉換結果儲存在stringstream物件的內部緩衝中。你不必擔心緩衝區溢位,因為這些物件會根據需要自動分配儲存空間。

你的編譯器支援嗎?

庫是最近才被列入c++標準的。(不要把與標準發布前被刪掉的弄混了。)因此,老一點 的編譯器,如gcc2.95,並不支援它。如果你恰好正在使用這樣的編譯器而又想使用的話,就要先對它進行公升級更新。

庫定義了三種類:istringstream、ostringstream和stringstream,分別用來進行流的輸入、輸出和輸入輸出操作。另 外,每個類都有乙個對應的寬字符集版本。簡單起見,我主要以stringstream為中心,因為每個轉換都要涉及到輸入和輸出操作。

注意,使用string物件來代替字元陣列。這樣可以避免緩衝區溢位的危險。而且,傳入引數和目標物件的型別被自動推導出來,即使使用了不正確的格式化符也沒有危險。

string到int的轉換

string result=」10000」;

int n=0;

stream<>n;//n等於10000

重複利用stringstream物件

如果你打算在多次轉換中使用同乙個stringstream物件,記住再每次轉換前要使用clear()方法;

在多次轉換中重複使用同乙個stringstream(而不是每次都建立乙個新的物件)物件最大的好處在於效率。stringstream物件的構造和析構函式通常是非常耗費cpu時間的。

在型別轉換中使用模板

你可以輕鬆地定義函式模板來將乙個任意的型別轉換到特定的目標型別。例如,需要將各種數字值,如int、long、double等等轉換成字串,要使用以乙個string型別和乙個任意值t 為引數的to_string()函式。to_string()函式將t 轉換為字串並寫入result中。使用str()成員函式來獲取流內部緩衝的乙份拷貝:

template

void to_string(string & result,const t& t)

ostringstream oss;//建立乙個流

oss}這樣,你就可以輕鬆地將多種數值轉換成字串了:

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

out_type convert(const in_value & t)

stringstream stream;

streamstream>>result;//向result中寫入值

return result;

這樣使用convert():

double d;

string salary;

string s=」12.56」;

d=convert(s);//d等於12.56

salary=convert(9000.0);//salary等於」9000」

結論

在過去留下來的程式**和純粹的c程式中,傳統的形式的轉換伴隨了我們很長的一段時間。但是,如文中所述,基於 stringstream的轉換擁有型別安全和不會溢位這樣搶眼的特性,使我們有充足得理由拋棄而使 用。庫還提供了另外乙個特性—可擴充套件性。你可以通過過載來支援自定義型別間的轉換。

一些例項:

stringstream通常是用來做資料轉換的。

相比c庫的轉換,它更加安全,自動和直接。

例子一:基本資料型別轉換例子 int轉string

#include 

#include 

#include 

int main() 

執行結果:

例子二:除了基本型別的轉換,也支援char *的轉換。

#include 

#include 

int main() 

例子三:再進行多次轉換的時候,必須呼叫stringstream的成員函式clear().

#include 

#include 

int main() 

執行clear的結果

沒有執行clear的結果

stringstream物件的使用

識別每行中的單詞 include include include using namespace std int main string line,word will hold a line and word from input respectively while getline cin,lin...

使用stringstream物件簡化型別轉換

c 標準庫中的提供了比ansi c的更高階的一些功能,即單純性 型別安全和可擴充套件性。在本文中,我將展示怎樣使用這些庫來實現安全和自動的型別轉換。為什麼要學習 如果你已習慣了風格的轉換,也許你首先會問 為什麼要花額外的精力來學習基於的型別轉換呢?也許對下面乙個簡單的例子的回顧能夠說服你。假設你想用...

使用stringstream物件簡化型別轉換

c 標準庫中的提供了比ansi c的更高階的一些功能,即單純性 型別安全和可擴充套件性。在本文中,我將展示怎樣使用這些庫來實現安全和自動的型別轉換。為什麼要學習 如果你已習慣了風格的轉換,也許你首先會問 為什麼要花額外的精力來學習基於的型別轉換呢?也許對下面乙個簡單的例子的回顧能夠說服你。假設你想用...