1. 程式人生 > >c++ 字串流 sstream(常用於格式轉換)

c++ 字串流 sstream(常用於格式轉換)

原文:http://blog.163.com/zhuandi_h/blog/static/180270288201291710222975/

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

例子一:基本資料型別轉換例子 int轉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"
} 

例子二:除了基本型別的轉換,也支援char *的轉換。
#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"
} 
 
例子三:再進行多次轉換的時候,必須呼叫stringstream的成員函式clear().
#include <sstream>
#include <iostream>
int main()
{
    std::stringstream stream;
    int first, second;
    stream<< "456"; //插入字串
    stream >> first; //轉換成int
    std::cout << first << std::endl;
    stream.clear(); //在進行多次轉換前,必須清除stream
    stream << true; //插入bool值
    stream >> second; //提取出int
    std::cout << second << std::endl;
} 
 

注:關於stream.clear()和stream.str(""),作用還不太清楚。又說clear是清除標誌位,str("")是清楚stream內容的。但在多次轉換過程是,的確是使用clear才準確,這是驗證過的。