如果你已習慣了風格的轉換,也許你首先會問:為什麼要花額外的精力來學習基於的型別 轉換呢?也許對下面乙個簡單的例子的回顧能夠說服你。假設你想用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)
這樣,你就可以輕鬆地將多種數值轉換成字串了:
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)
這樣使用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的用法
1.利用輸入輸出做資料轉換
stringstream ss_stream;
ss_stream << i; // 將int輸入流中
ss_stream >> str; // 將ss_stream中的數值輸出到str中
//注意:如果做多次資料轉換;必須呼叫clear()來設定轉換模式
ss_stream << "456";
ss_stream >> i; // 首先將字串轉換為int
ss_stream.clear();
ss_stream << true;
ss_stream >> i; // 然後將bool型轉換為int;假如之前沒有做clear,那麼i會出錯
//執行clear的結果
i = 456
i = 1
//沒有執行clear的結果
i = 456
i = 8800090900
2.支援char*的輸入和輸出char sz_buf[20];
ss_stream << 8888;
ss_stream >> sz_buf; // 直接將數輸出到sz_buf字元陣列中
3.來儲存可變資料的列表stringstream ss_stream;ss_stream << "字串一"
<< endl;
ss_stream << "字串二"
<< endl;
ss_stream << "字串三"
<< endl;
ss_stream << "字串四"
<< endl;
ss_stream << "字串五"
<< endl;
char buffer[100];
while ( ss_stream.getline(buffer, sizeof(buffer))
ss_stream("");// 釋放字串流中的資源
// 或者用string來接收
stringstream ss_stream;
string stemp;
while ( getline(ss_stream, stemp) )
STL之演算法
演算法是指解決問題的方 而完整的描述,對於規範的輸入,在有限時間內要獲得所需要的輸出。不同的演算法可能使用不同的時間 空間或效率完成同樣的任務。想要評估乙個演算法的好壞,目前可以通過時間複雜度和空間複雜度來進行衡量。時間複雜度,是指演算法執行指令所需的計算量。演算法的執行時間和其所要處理的資料之間存...
STL之函式物件
目錄 為了使類屬演算法具有靈活性,stl常使用函式的過載機制為演算法提供兩種形式。演算法的第一種形式使用的是常規的操作來實現。第二種形式中,演算法可以根據使用者指定的準測對元素經行處理。函式物件包含了乙個可以通過函式呼叫運算子 使用的函式。實際上,函式物件是過載了函式呼叫運算子operator 的類...
STL之堆操作
首先來看完全二叉樹的定義 若設二叉樹的深度為h,除第 h 層外,其它各層 1 h 1 的結點數都達到最大個數,第 h 層所有的結點都連續集中在最左邊,這就是完全二叉樹。而將一維陣列視為完全二叉樹書得到的即為堆。堆效率極高,像十分常用的 排序演算法 dijkstra 演算法 prim 演算法等都要用堆...