C++常用資料型別轉換詳解
<sstream>庫定義了三種類:istringstream、ostringstream和stringstream,分別用來進 行流的輸入、輸出和輸入輸出操作。另外,每個類都有一個對應的寬字符集版本。簡單起見,我主要以stringstream為中心,因為每個轉換都要涉及到 輸入和輸出操作。示例1示範怎樣使用一個stringstream物件進行從
string到int型別的轉換
注意,<sstream>使用string物件來代替字元陣列。這樣可以避免緩衝區溢位的危險。而且,傳入引數和目標物件的型別被自動推匯出來,即使使用了不正確的格式化符也沒有危險。
示例1:
std::stringstream stream;
string result="10000";
int n = 0;
stream << result;
stream >> n;//n等於10000
int到string型別的轉換
string result;
int n = 12345;
stream << n;
result =stream.str();// result等於"12345"
重複利用stringstream物件
如果你打算在多次轉換中使用同一個stringstream物件,記住再每次轉換前要使用clear()方法,在多次轉換中重複使用同一個 stringstream(而不是每次都建立一個新的物件)物件最大的好處在於效率。stringstream物件的構造和解構函式通常是非常耗費CPU 時間的。經試驗,單單使用clear()並不能清除stringstream物件的內容,僅僅是了該物件的狀態,要重複使用同一個 stringstream物件,需要使用str()重新初始化該物件。
示例2:
std::stringstream strsql;
for (int i= 1; i < 10; ++i)
{
strsql << "insert into test_tab values(";
strsql << i << ","<< (i+10) << ");";
std::string str = strsql.str(); // 得到string
res = sqlite3_exec(pDB,str.c_str(),0,0, &errMsg);
std::cout << strsql.str() << std::endl;
strsql.clear();
strsql.str("");
}
轉換中使用模板
也可以輕鬆地定義函式模板來將一個任意的型別轉換到特定的目標型別。例如,需要將各種數字值,如int、long、double等等轉換成字串, 要使用 以一個string型別和一個任意值t為引數的to_string()函式。to_string()函式將t轉換為字串並寫入result中。使用 str()成員函式來獲取流內部緩衝的一份拷貝:
示例3:
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>。
當然現在還有一個更好的選擇,那就是使用boost庫中的lexical_cast,它是型別安全的轉換。如下例:
#include <iostream>
#include <sstream>
#include <string>
#include <cstdlib>
#include <boost/lexical_cast.hpp>
using namespace std;
using namespace boost;
int main(void)
try{
// 以下是內建型別向 string 轉換的解決方案
//lexical_cast 優勢明顯
int ival;
char cval;
ostringstream out_string;
string str0;
string str1;
ival = 100;
cval = 'w';
out_string << ival << " " << cval;
str0 = out_string.str();
str1 = lexical_cast<string>(ival)
+ lexical_cast<string>(cval);
cout << str0 << endl;
cout << str1 << endl;
// 以下是 string 向內建型別轉換的解決方案
// 幾乎和 stringstrem 相比,lexical_cast就是型別安全的,
int itmpe;
char ctmpe;
str0 = "100k";
str1 = "100h";
istringstream in_string( str0 );
in_string >> itmpe >> ctmpe;
cout << itmpe << " " << ctmpe << endl;
itmpe = lexical_cast<int>(str1);
ctmpe = lexical_cast<char>(str1);
system( "PAUSE" );
return 0;
}
catch(bad_lexical_cast e)
{
cout << e.what() << endl;
cin.get();
}