一、string
string 是 C++ 提供的字串型態,和 C 的字串相比,除了有不限長度的優點外,還有其他許多友善的功能。要使用 string, 必須先加入這一行:接下來要宣告一個字串變量,可以寫成:<span style="background-color: rgb(255, 255, 255);"><span style="font-size:18px;">#include <string></span></span>
我們也可以在宣告的同時讓它設成某個字串:<span style="background-color: rgb(255, 255, 255);"><span style="font-size:18px;">string s;</span></span>
<span style="background-color: rgb(255, 255, 255);"><span style="font-size:18px;">string s="TCGS";</span></span>
而要取得其中某一個字元,和傳統C 的字串一樣是用 s[i] 的方式取得。比較不一樣的是如果 s 有三個字元,傳統 C 的字串的 s[3] 是 0 字元,但是 C++ 的 string 則是隻到 s[2] 這個字元而已。
下面我們把 string 與 字元陣列的語法做一個對照:
從上面的表格,我們可以發現 string 的用法比較直覺,是以如果沒有特別的需要,盡量使用 string 會比較友善。
操作 string 字元陣列 宣告字串 string s; char s[100]; 取得第 i 個字元 s[i] s[i] 字串長度 s.length()
或 s.size()
strlen(s) 讀取一行 getline(cin, s); gets(s); 設成某字串 s="TCGS"; strcpy(s, "TCGS"); 字串相加 s=s+"TCGS";
s+="TCGS"
strcat(s, "TCGS"); 字串比較 s=="TCGS" strcmp(s, "TCGS")
二、stringstream
stringstream 是 C++ 提供的另一個字串型的串流(stream)物件,和之前學過的 iostream、fstream 有類似的操作方式。要使用 stringstream, 必須先加入這一行:stringstream 主要是用在將一個字串分割,可以先用 clear( )以及 str( ) 將指定字串設定成一開始的內容,再用 >> 把個別的資料輸出,例如:<span style="background-color: rgb(255, 255, 255);"><span style="font-size:18px;">#include <sstream></span></span>
<span style="background-color: rgb(255, 255, 255);"><span style="font-size:18px;">string s; stringstream ss; int a, b, c; getline(cin, s); ss.clear(); ss.str(s); ss >> a >> b >> c;</span></span>
下面我們看到一個使用 stringstream 的例子:
題目:輸入的第一行有一個數字 N 代表接下來有 N 行資料,每一行資料裡有不固定個數的整數(最多 20 個,每行最大 200 個字元),請你寫一個程式將每行的總和印出來。
輸入:
3
1 2 3
20 17 23 54 77 60
111 222 333 444 555 666 777 888 999
輸出:
6
251
4995
程式如下:
<span style="background-color: rgb(255, 255, 255);"><span style="font-size:18px;">string s; stringstream ss; int n, i, sum, a; cin >> n; getline(cin, s); // 讀取換行 for (i=0; i<n; i++) { getline(cin, s); ss.clear(); ss.str(s); sum=0; while (1) { ss >> a; if ( ss.fail() ) break; sum+=a; } cout << sum << endl; }</span></span>
string類型是c語言中char *類型的一種更便利的實作。使用這個類型,不用再去刻意考慮記憶體的事兒。在做快速開發的時候,string對象提供的便利,還是相當出色的。然而,在這兒提醒一下:string類型很有可能成為一個工程效率問題的根源,産品級别的應用當中,應該盡量避免在深層循環嵌套中使用string類型。
除size()外,另外兩個string常用的方法是find和substr。在下面的代碼當中:
string str = "aaaaddddssdfsasdf";
size_t pos = str.find("ssdf", 3); //用if(pos == string::npos) 用來判斷是否找到子串。
string str2 = str.substr(pos, 5);
find函數從str的第3個位置查起,找到ssdf這個子串後,傳回子串的位置。而substr函數從pos位置開始,截取5個字元,指派給str2。也就是說,str2之後的内容将是ssdfs。
stringstream是字元串流,經常被我用來作資料切分或者類型轉化。一個經常被我用到的函數如下:
string i2s(int i, int len = 0)
{
stringstream ss;
ss << setw(len) << setfill('0') << i;
return ss.str():
}
以i2s(7, 3)形式調用這個函數,傳回的結果是字元串007。我通常在循環裡,這樣産生或者周遊一些檔案。
使用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; } 運作clear的結果 沒有運作clear的結果 |