1. 程式人生 > >C++ string詳解

C++ string詳解

還有必要再重複一下C++字串和C字串轉換的問題,許多人會遇到這樣的問題,自己做的程式要呼叫別人的函式、類什麼的(比如資料庫連線函式Connect(char*,char*)),但別人的函式引數用的是char*形式的,而我們知道,c_str()、data()返回的字元陣列由該字串擁有,所以是一種const char*,要想作為上面提及的函式的引數,還必須拷貝到一個char*,而我們的原則是能不使用C字串就不使用。那麼,這時候我們的處理方式是:如果此函式對引數(也就是char*)的內容不修改的話,我們可以這樣Connect((char*)UserID.c_str(), (char*)PassWD.c_str()),但是這時候是存在危險的,因為這樣轉換後的字串其實是可以修改的(有興趣地可以自己試一試),所以我強調除非函式呼叫的時候不對引數進行修改,否則必須拷貝到一個char*上去。當然,更穩妥的辦法是無論什麼情況都拷貝到一個char*上去。同時我們也祈禱現在仍然使用C字串進行程式設計的高手們(說他們是高手一點兒也不為過,也許在我們還穿開襠褲的時候他們就開始程式設計了,哈哈…)寫的函式都比較規範,那樣我們就不必進行強制轉換了。

2.3元素存取
    我們可以使用下標操作符[]和函式at()對元素包含的字元進行訪問。但是應該注意的是操作符[]並不檢查索引是否有效(有效索引0~str.length()),如果索引失效,會引起未定義的行為。而at()會檢查,如果使用at()的時候索引無效,會丟擲out_of_range異常。
    有一個例外不得不說,const string a;的操作符[]對索引值是a.length()仍然有效,其返回值是’/0’。其他的各種情況,a.length()索引都是無效的。舉例如下:
const string Cstr(“const string”);
string Str(“string”);

Str[3];    //ok
Str.at(3);  //ok

Str[100]; //未定義的行為
Str.at(100);  //throw out_of_range

Str[Str.length()]  //未定義行為
Cstr[Cstr.length()] //返回 ‘/0’
Str.at(Str.length());//throw out_of_range
Cstr.at(Cstr.length()) ////throw out_of_range

我不贊成類似於下面的引用或指標賦值:
char& r=s[2];
char* p= &s[3];
因為一旦發生重新分配,r,p立即失效。避免的方法就是不使用。

2.4比較函式
    C++字串支援常見的比較操作符(>,>=,<,<=,==,!=),甚至支援string與C-string的比較(如 str<”hello”)。在使用>,>=,<,<=這些操作符的時候是根據“當前字元特性”將字元按字典順序進行逐一得比較。字典排序靠前的字元小,比較的順序是從前向後比較,遇到不相等的字元就按這個位置上的兩個字元的比較結果確定兩個字串的大小。同時,string(“aaaa”) <string(aaaaa)。
    另一個功能強大的比較函式是成員函式compare()。他支援多引數處理,支援用索引值和長度定位子串來進行比較。他返回一個整數來表示比較結果,返回值意義如下:0-相等 〉0-大於 <0-小於。舉例如下:
    string s(“abcd”);
    
    s.compare(“abcd”); //返回0
    s.compare(“dcba”); //返回一個小於0的值
    s.compare(“ab”); //返回大於0的值
    
s.compare(s); //相等
    s.compare(0,2,s,2,2); //用”ab”和”cd”進行比較 小於零
    s.compare(1,2,”bcx”,2); //用”bc”和”bc”比較。
怎麼樣?功能夠全的吧!什麼?還不能滿足你的胃口?好吧,那等著,後面有更個性化的比較演算法。先給個提示,使用的是STL的比較演算法。什麼?對STL一竅不通?靠,你重修吧!

2.5 更改內容
這在字串的操作中佔了很大一部分。
首先講賦值,第一個賦值方法當然是使用操作符=,新值可以是string(如:s=ns) 、c_string(如:s=”gaint”)甚至單一字元(如:s=’j’)。還可以使用成員函式assign(),這個成員函式可以使你更靈活的對字串賦值。還是舉例說明吧:
s.assign(str); //不說
s.assign(str,1,3);//如果str是”iamangel” 就是把”ama”賦給字串
s.assign(str,2,string::npos);//把字串str從索引值2開始到結尾賦給s
s.assign(“gaint”); //不說
s.assign(“nico”,5);//把’n’ ‘I’ ‘c’ ‘o’ ‘/0’賦給字串
s.assign(5,’x’);//把五個x賦給字串
把字串清空的方法有三個:s=””;s.clear();s.erase();(我越來越覺得舉例比說話讓別人容易懂!)。
string提供了很多函式用於插入(insert)、刪除(erase)、替換(replace)、增加字元。
先說增加字元(這裡說的增加是在尾巴上),函式有 +=、append()、push_back()。舉例如下:
s+=str;//加個字串
s+=”my name is jiayp”;//加個C字串
s+=’a’;//加個字元

s.append(str);
s.append(str,1,3);//不解釋了 同前面的函式引數assign的解釋
s.append(str,2,string::npos)//不解釋了

s.append(“my name is jiayp”);
s.append(“nico”,5);
s.append(5,’x’);

s.push_back(‘a’);//這個函式只能增加單個字元 對STL熟悉的理解起來很簡單

也許你需要在string中間的某個位置插入字串,這時候你可以用insert()函式,這個函式需要你指定一個安插位置的索引,被插入的字串將放在這個索引的後面。
    s.insert(0,”my name”);
    s.insert(1,str);
這種形式的insert()函式不支援傳入單個字元,這時的單個字元必須寫成字串形式(讓人噁心)。既然你覺得噁心,那就不得不繼續讀下面一段話:為了插入單個字元,insert()函式提供了兩個對插入單個字元操作的過載函式:insert(size_type index,size_type num,chart c)和insert(iterator pos,size_type num,chart c)。其中size_type是無符號整數,iterator是char*,所以,你這麼呼叫insert函式是不行的:insert(0,1,’j’);這時候第一個引數將轉換成哪一個呢?所以你必須這麼寫:insert((string::size_type)0,1,’j’)!第二種形式指出了使用迭代器安插字元的形式,在後面會提及。順便提一下,string有很多操作是使用STL的迭代器的,他也儘量做得和STL靠近。
刪除函式erase()的形式也有好幾種(真煩!),替換函式replace()也有好幾個。舉例吧:
string s=”il8n”;
s.replace(1,2,”nternationalizatio”);//從索引1開始的2個替換成後面的C_string
s.erase(13);//從索引13開始往後全刪除
s.erase(7,5);//從索引7開始往後刪5個

2.6提取子串和字串連線

題取子串的函式是:substr(),形式如下:
s.substr();//返回s的全部內容
s.substr(11);//從索引11往後的子串
s.substr(5,6);//從索引5開始6個字元
把兩個字串結合起來的函式是+。(誰不明白請致電120)

2.7輸入輸出操作
1.>> 從輸入流讀取一個string。
2.<< 把一個string寫入輸出流。
另一個函式就是getline(),他從輸入流讀取一行內容,直到遇到分行符或到了檔案尾。

2.8搜尋與查詢
查詢函式很多,功能也很強大,包括了:
    find()
    rfind()
    find_first_of()
    find_last_of()
    find_first_not_of()
    find_last_not_of()
這些函式返回符合搜尋條件的字元區間內的第一個字元的索引,沒找到目標就返回npos。所有的函式的引數說明如下:
第一個引數是被搜尋的物件。第二個引數(可有可無)指出string內的搜尋起點索引,第三個引數(可有可無)指出搜尋的字元個數。比較簡單,不多說 不理解的可以向我提出,我再仔細的解答。當然,更加強大的STL搜尋在後面會有提及。
最後再說說npos的含義,string::npos的型別是string::size_type,所以,一旦需要把一個索引與npos相比,這個索引值必須是string::size)type型別的,更多的情況下,我們可以直接把函式和npos進行比較(如:if(s.find(“jia”)==string::npos))。
    第二部分是關於C++字串對迭代器的支援的,視大家的需要我將寫出來(意思就是不需要就算了,我樂得輕省,哈哈…)。
好了,大概的對string型別進行了闡述,希望起到拋磚引玉的作用,讓初學者對string有個瞭解而不必已開始就面對複雜的內部結構和無數個注意事項。對字串更詳細地講解有很多參考書,其實我的內容也是從C++標準程式庫得來的,加上幾句自己的看法,所以要感謝這本書的作者和譯者。任何人對本文進行引用都要標明作者是Nicolai M.Josuttis 譯者是侯捷/孟巖。不過不要提及我,任何觀點的錯誤都與我無關(除了這裡邊體現我主觀想法的幾句話,也就那幾句話)。

相關推薦

c++ string2 assign

assign方法可以理解為先將原字串清空,然後賦予新的值作替換。 返回型別為 string型別的引用。其常用的過載也有下列幾種: a. string& assign ( const string& str ); 將str替換原字串的內容 舉例: strin

C++ string

還有必要再重複一下C++字串和C字串轉換的問題,許多人會遇到這樣的問題,自己做的程式要呼叫別人的函式、類什麼的(比如資料庫連線函式Connect(char*,char*)),但別人的函式引數用的是char*形式的,而我們知道,c_str()、data()返回的字元陣列由該字串擁有,所以是一種const cha

c++string

string作為一個類出現,它整合的操作函式足以完成我們大多數情況下(甚至是100%)的需要,我們可以用=進行賦值操作,==進行比較,+做連線。可以把他看成c++基本資料型別。1.標頭檔案#include<string>2宣告一個c++字串:string str;

C++字串 string

      之所以拋棄char*的字串而選用C++標準程式庫中的string類,是因為他和前者比較起來,不必擔心記憶體是否足夠、字串長度等等,而且作為一個泛型類出現,他整合的操作函式足以完成我們大多數情況下(甚至是100%)的需要。我們可以用 = 進行賦值操作,== 進

C++中的string

標準庫型別string表示可變長的字元序列,為了在程式中使用string型別,我們必須包含標頭檔案: #include <string>  宣告一個字串 宣告一個字串有很多種方式,具體如下: 1 string s;//呼叫預設建構函式,s為一個空

C# LINQ From Where Select Group Into OrderBy Let Join

分享 str 關聯 例如 數據 lln ole inf emp 目錄 1. 概述 2. from子句 3. where子句 4. select子句 5. group子句 6. into子句 7. 排序子句 8. let子句 9. join子句 10. 小結 1. 概述

C++ 模板(二)(轉)

創建 規則 error ++ 例如 public err iostream () 四、類模板的默認模板類型形參   1、可以為類模板的類型形參提供默認值,但不能為函數模板的類型形參提供默認值。函數模板和類模板都可以為模板的非類型形參提供默認值。   2、類模板的類型形

C-pthread_cond_wait

read thread 一個 同時 多個 wait 產生 adc 全部 pthread_cond_wait() 用於阻塞當前線程,等待別的線程使用 pthread_cond_signal() 或 pthread_cond_broadcast 來喚醒它。 pthread_co

C#】屬性

radi 們的 cal com 位數 struct serializa 表達式 font 目錄結構: contents structure [+] 屬性和字段的區別 無參屬性 自動實現的屬性 對象和集合初始化器 匿名類型 System.Tu

C# DataTable

lec row 條件 而且 獲取 int mar 索引 typeof 添加引用 using System.Data; 創建表 //創建一個空表 DataTable dt = new DataTable(); //創建一個名為"Table_New"的空表 DataT

廣義表C/C++實現

所謂的廣義表就是單鏈表的擴充套件,就是節點可以是一個元素也可以是一個連結串列。官方的說法是,原子節點和子表節點。廣義表用遞迴的方法來建立是最簡單易懂的,也符合廣義表的思想。 C程式碼實現下載 C++程式碼實現下載 (備用下載地址 ) 1.廣義表的各種形態 (1)、A = (

C++ 類

1、類的定義 class 類名稱 { public:      公有成員(外部介面,可被使用該類的所有程式碼所使用) private:      私有成員 (只允許本類中的函式訪問,而類外部的任何

C#反射

ron new t getc 給定 types console const property rom http://blog.csdn.net/educast/article/details/2894892(轉) 兩個現實中的例子:1、B超:大家體檢的時候大概都做過B超吧,

PowerShell 下的 grep —— Select-String 及二者對比

使用 help Select-String -ShowWindow 命令我們可以很容易的得到一份中文的幫助文件,其中有這樣兩段話: Select-String cmdlet 在輸入字串和檔案中搜索文字和文字模式。你可以像使用 UNIX 中的 Grep 和 Windo

C# 特性(Attribute)

今天整理關於特性的資料,之前都忘了,今天整理一下 參考《C#高階程式設計》第10版 0X01 特性(Attribute) 特性定義 特性不會影響編譯過程,因為編譯器不能識別它們,但這些特性在應用於程式元素

C++容器

什麼是容器 首先,我們必須理解一下什麼是容器,在C++ 中容器被定義為:在資料儲存上,有一種物件型別,它可以持有其它物件或指向其它對像的指標,這種物件型別就叫做容器。很簡單,容器就是儲存其它物件的對 象,當然這是一個樸素的理解,這種“物件”還包含了一系列處理“其它物件”的方法,因為這些方法在程式

thinking in java (三十三) ----- String

String介紹 String是java中的字串,繼承於CharSequence。 String類包含的API介面非常多,我們對其進行了分類,並且都給出了演示程式。 String個CharSequence關係 String繼承於CharSequence,也就是

Tcl/Tk string

分享一下我老師大神的人工智慧教程!零基礎,通俗易懂!http://blog.csdn.net/jiangjunshow 也歡迎大家轉載本篇文章。分享知識,造福人民,實現我們中華民族偉大復興!        

C/C++陣列(一維和二維)

陣列這東西,說說都懂,但是似乎並沒有完全吃透,導致很多地方有疑惑。所以再梳理一遍。   陣列定義 陣列是存放型別相同的物件的容器,這些物件本身沒有名字,需要通過其所在位置訪問。 從定義中可以看出,陣列存放的是物件且型別相同。所以不存在引用的物件(引用不是物件)

哈夫曼樹C++實現

哈夫曼樹的介紹 Huffman Tree,中文名是哈夫曼樹或霍夫曼樹,它是最優二叉樹。 定義:給定n個權值作為n個葉子結點,構造一棵二叉樹,若樹的帶權路徑長度達到最小,則這棵樹被稱為哈夫曼樹。 這個定義裡面涉及到了幾個陌生的概念,下面就是一顆哈夫曼樹,我們來看圖解答。 (01) 路徑和路徑長度