C++ string(C++字串)詳解

2020-07-16 10:04:23
用字元陣列存放字串容易發生陣列越界的錯誤,而且往往難以察覺。因此,C++ 標準模板庫設計了 string 資料型別,專門用於字串處理。

string 型別的變數就是用來存放字串的,也叫“string物件”。string 並不是 C++ 的基本資料型別,它是 C++ 標準模板庫中的一個“類”。關於這一點,現在不必深究,以後會學到。這裡只要學會如何使用 string 物件即可。

要使用 string 物件,必須包含標頭檔案 <string>。

在用 C++ 程式設計時,要優先考慮用 string 物件來處理字串,因為其用法比字元陣列更簡單,而且不容易出錯。

定義 string 物件

定義 string 物件的方法和定義普通變數沒有區別,就是:

string 變數名;

而且定義時還可以初始化,例如:
string str1;  //定義 string 物件 str1
string city = "Beijing";  //定義 string 物件 city,並對其初始化
定義 string 物件時,如果不對其初始化,則其值是空串,即""

與字元陣列不同的是,一個 string 物件的大小是固定的,即表示式 sizeof(string) 的值是固定的,和其中存放的字串的長度無關。

但這個固定的值在不同編譯器中並不相同。例如,在 Dev C++ 中是 4,在 Visual Studio 2010 中是 32。string 物件中並不直接存放字串,字串會在別處開闢記憶體空間存放,string 物件中只存放該記憶體空間的地址,或者再加上其他一些資訊。

還可以定義 string 物件陣列,例如:
string as [] = {"Beijing", "Shanghai", "Chengdu"};
cout << as[1];  //輸出 Shanghai

string物件的輸入輸出

string 物件同樣可以用 cin、cout 進行輸入和輸出,例如:
string s1, s2;
cin >> s1 >> s2;
cout << s1 << "," << s2;

string物件的賦值

string 物件之間可以互相賦值,也可以用字串常數和字元陣列的名字對 string 物件進行 賦值。賦值時不需要考慮被賦值的物件是否有足夠空間存放字串的問題。例如:
string s1, s2 = "ok";
s1 = "China";
s2 = s1;  //s1和s2不等長也沒關係,賦值後s2內容和s1相同
char name [] = "Lady Gaga";
s1 = name;  //賦值後s1中的內容和name相同,修改s1不會影響name
用普通字串對 string 物件賦值,普通字串的內容會被複製到 string 物件所管理的那片記憶體空間中。

string物件的運算

string 物件之間可以用 <、<=、==、>=、> 運算子進行比較,還可以用+將兩個 string 物件相加、將一個字串常數和 string 物件相加、將一個字元陣列和 string 物件相加,相當於進行字串連線。+=運算子也適用於 string 物件。此外,string 物件還可以通過[]運算子和下標存取字串中的某個字元。例如:
string s1 = "123", s2 = "abc11", s3;  //s3是空串
s3 = sl + s2;  //s3 變成"123abc"
s3 += "de";  //s3 變成"123abcde"
bool b = s1 < s3;  //b 為 true
char c = s1[2];  //c變成'3'(下標從0開始計算)
s1[2] = '5';  //s1 變成”125"
string 物件在比較大小時是按詞典序比較的,而且是大小寫相關的。由於大寫字母的 ASCII 碼小於小寫字母的 ASCII 碼('A' ~ 'Z' 的 ASCII 碼是 0x41 ~ 0x5a,'a' ~ 'z' 的 ASCII 碼是 0x61 ~ 0x7a),所以 Zbc 比 abc 小。

string物件用法範例

string 物件還有一些“成員函數”,可以用來很方便地實現一些功能,如查詢子串等。這些成員函數的呼叫方法就是“string物件名.成員函數名”。

具體的成員函數我們將在後邊介紹。這裡給出一個 string 物件的基本用法範例。
#include <iostream>
#include <string>  //要使用string物件,必須包含此標頭檔案
using namespace std;
int main()
{
    string s1 = "123", s2;  //s2 是空串
    s2 += s1;  //s2 ="123"
    s1 = "abc";  //s1 = "abc"
    s1 += "def";  //s1 = "abcdef"
    cout << "1)" << s1 << endl;  //輸出 1)abcdef
    if (s2 < s1)
        cout << "2)s2 < s1" << endl;  //輸出 2)s2<s1
    else
        cout << "2)s2 >= s1" << endl;  //不被執行

    s2[1] = 'A';  //s2 ="1A3"
    s1 = "XYZ" + s2;  //s1 = "XYZlA3"
    string s3 = s1 + s2;  //s3 = "XYZ1A31A3"
    cout << "3)" << s3 << endl;  //輸出 3)XYZ1A31A3
    cout << "4)" << s3.size() << endl;  //求s3長度,輸出 4)9
    string s4 = s3.substr(1, 3);  //求s3從下標1開始,長度為3的子串
    cout << "5)" << s4 << endl;  //輸出 5)YZ1
    char str[20];
    strcpy(str, s4.c_str());  //複製 s4 中的字串到 str
    cout << "6)" << str << endl;  //輸出 6)YZ1
    return 0;
}
第 19 行和第 20 行分別呼叫了 string 物件的成員函數 size 和 substr,用於求 s3 的長度和子串。

第 23 行,通過呼叫 c_str 成員函數,能將 s4 的內容複製到 str。