1. 程式人生 > >c++vector用法詳解

c++vector用法詳解

     用法:

          1.檔案包含:     

           首先在程式開頭處加上#include<vector>以包含所需要的類檔案vector

          還有一定要加上using namespace std;

          2.變數宣告:

               2.1 例:宣告一個int向量以替代一維的陣列:vector <int> a;(等於聲明瞭一個int陣列a[],大小沒有指定,可以動態的向裡面新增刪除)。

               2.2 例:用vector代替二維陣列.其實只要宣告一個一維陣列向量即可,而一個數組的名字其實代表的是它的首地址,所以只要宣告一個地址的向量即可,即:vector <int *> a.同理想用向量代替三維陣列也是一樣,vector <int**>a;再往上面依此類推.

          3.具體的用法以及函式呼叫:

          3.1 如何得到向量中的元素?其用法和陣列一樣:

          例如:

          vector <int *> a

          int b = 5;

          a.push_back(b);//該函式下面有詳解

          cout<<a[0];       //輸出結果為5

1.push_back   在陣列的最後新增一個數據
2.pop_back    去掉陣列的最後一個數據 
3.at                得到編號位置的資料
4.begin           得到陣列頭的指標
5.end             得到陣列的最後一個單元+1的指標
6.front        得到陣列頭的引用
7.back            得到陣列的最後一個單元的引用
8.max_size     得到vector最大可以是多大
9.capacity       當前vector分配的大小
10.size           當前使用資料的大小
11.resize         改變當前使用資料的大小,如果它比當前使用的大,者填充預設值
12.reserve      改變當前vecotr所分配空間的大小
13.erase         刪除指標指向的資料項
14.clear          清空當前的vector
15.rbegin        將vector反轉後的開始指標返回(其實就是原來的end-1)
16.rend          將vector反轉構的結束指標返回(其實就是原來的begin-1)
17.empty        判斷vector是否為空
18.swap         與另一個vector交換資料

         3.2  詳細的函式實現功能:其中vector<int> c.

                             c.clear()         移除容器中所有資料。

                             c.empty()         判斷容器是否為空。

                             c.erase(pos)        刪除pos位置的資料

                             c.erase(beg,end) 刪除[beg,end)區間的資料

                             c.front()         傳回第一個資料。

                             c.insert(pos,elem)  在pos位置插入一個elem拷貝

                             c.pop_back()     刪除最後一個數據。

                             c.push_back(elem) 在尾部加入一個數據。

                             c.resize(num)     重新設定該容器的大小

                             c.size()         回容器中實際資料的個數。

                             c.begin()           返回指向容器第一個元素的迭代器

                             c.end()             返回指向容器最後一個元素的迭代器


               4.記憶體管理與效率

      1》使用reserve()函式提前設定容量大小,避免多次容量擴充操作導致效率低下。

        關於STL容器,最令人稱讚的特性之一就是是隻要不超過它們的最大大小,它們就可以自動增長到足以容納你放進去的資料。(要知道這個最大值,只要呼叫名叫max_size的成員函式。)對於vector和string,如果需要更多空間,就以類似realloc的思想來增長大小。vector容器支援隨機訪問,因此為了提高效率,它內部使用動態陣列的方式實現的。在通過 reserve() 來申請特定大小的時候總是按指數邊界來增大其內部緩衝區。當進行insert或push_back等增加元素的操作時,如果此時動態陣列的記憶體不夠用,就要動態的重新分配當前大小的1.5~2倍的新記憶體區,再把原陣列的內容複製過去。所以,在一般情況下,其訪問速度同一般陣列,只有在重新分配發生時,其效能才會下降。正如上面的程式碼告訴你的那樣。而進行pop_back操作時,capacity並不會因為vector容器裡的元素減少而有所下降,還會維持操作之前的大小。對於vector容器來說,如果有大量的資料需要進行push_back,應當使用reserve()函式提前設定其容量大小,否則會出現許多次容量擴充操作,導致效率低下。

      reserve成員函式允許你最小化必須進行的重新分配的次數,因而可以避免真分配的開銷和迭代器/指標/引用失效。但在我解釋reserve為什麼可以那麼做之前,讓我簡要介紹有時候令人困惑的四個相關成員函式。在標準容器中,只有vector和string提供了所有這些函式。

(1) size()告訴你容器中有多少元素。它沒有告訴你容器為它容納的元素分配了多少記憶體。 
(2) capacity()告訴你容器在它已經分配的記憶體中可以容納多少元素。那是容器在那塊記憶體中總共可以容納多少元素,而不是還可以容納多少元素。如果你想知道一個vector或string中有多少沒有被佔用的記憶體,你必須從capacity()中減去size()。如果size和capacity返回同樣的值,容器中就沒有剩餘空間了,而下一次插入(通過insert或push_back等)會引發上面的重新分配步驟。
(3) resize(Container::size_type n)強制把容器改為容納n個元素。呼叫resize之後,size將會返回n。如果n小於當前大小,容器尾部的元素會被銷燬。如果n大於當前大小,新預設構造的元素會新增到容器尾部。如果n大於當前容量,在元素加入之前會發生重新分配。
(4) reserve(Container::size_type n)強制容器把它的容量改為至少n,提供的n不小於當前大小。這一般強迫進行一次重新分配,因為容量需要增加。(如果n小於當前容量,vector忽略它,這個呼叫什麼都不做,string可能把它的容量減少為size()和n中大的數,但string的大小沒有改變。在我的經驗中,使用reserve來從一個string中修整多餘容量一般不如使用“交換技巧”,那是條款17的主題。)

     這個簡介表示了只要有元素需要插入而且容器的容量不足時就會發生重新分配(包括它們維護的原始記憶體分配和回收,物件的拷貝和析構和迭代器、指標和引用的失效)。所以,避免重新分配的關鍵是使用reserve儘快把容器的容量設定為足夠大,最好在容器被構造之後立刻進行。

例如,假定你想建立一個容納1-1000值的vector<int>。沒有使用reserve,你可以像這樣來做:

vector<int> v;
for (int i = 1; i <= 1000; ++i) v.push_back(i);
在大多數STL實現中,這段程式碼在迴圈過程中將會導致2到10次重新分配。(10這個數沒什麼奇怪的。記住vector在重新分配發生時一般把容量翻倍,而1000約等於210。)

把程式碼改為使用reserve,我們得到這個:

vector<int> v;
v.reserve(1000);
for (int i = 1; i <= 1000; ++i) v.push_back(i);
這在迴圈中不會發生重新分配。

在大小和容量之間的關係讓我們可以預言什麼時候插入將引起vector或string執行重新分配,而且,可以預言什麼時候插入會使指向容器中的迭代器、指標和引用失效。例如,給出這段程式碼,

string s;
...
if (s.size() < s.capacity()) {
s.push_back('x');
}
push_back的呼叫不會使指向這個string中的迭代器、指標或引用失效,因為string的容量保證大於它的大小。如果不是執行push_back,程式碼在string的任意位置進行一個insert,我們仍然可以保證在插入期間沒有發生重新分配,但是,與伴隨string插入時迭代器失效的一般規則一致,所有從插入位置到string結尾的迭代器/指標/引用將失效。

回到本條款的主旨,通常有兩情況使用reserve來避免不必要的重新分配。第一個可用的情況是當你確切或者大約知道有多少元素將最後出現在容器中。那樣的話,就像上面的vector程式碼,你只是提前reserve適當數量的空間。第二種情況是保留你可能需要的最大的空間,然後,一旦你新增完全部資料,修整掉任何多餘的容量。

       2》使用“交換技巧”來修整vector過剩空間/記憶體

      有一種方法來把它從曾經最大的容量減少到它現在需要的容量。這樣減少容量的方法常常被稱為“收縮到合適(shrink to fit)”。該方法只需一條語句:vector<int>(ivec).swap(ivec);
表示式vector<int>(ivec)建立一個臨時vector,它是ivec的一份拷貝:vector的拷貝建構函式做了這個工作。但是,vector的拷貝建構函式只分配拷貝的元素需要的記憶體,所以這個臨時vector沒有多餘的容量。然後我們讓臨時vector和ivec交換資料,這時我們完成了,ivec只有臨時變數的修整過的容量,而這個臨時變數則持有了曾經在ivec中的沒用到的過剩容量。在這裡(這個語句結尾),臨時vector被銷燬,因此釋放了以前ivec使用的記憶體,收縮到合適。

     3》用swap方法強行釋放STL Vector所佔記憶體

template < class T> void ClearVector( vector<T>& v )

    vector<T>vtTemp;
    vtTemp.swap( v );

如 
    vector<int> v ;
    nums.push_back(1);
    nums.push_back(3);
    nums.push_back(2);
    nums.push_back(4);
    vector<int>().swap(v);

/* 或者v.swap(vector<int>()); */

/*或者{ std::vector<int> tmp = v;   v.swap(tmp);   }; //加大括號{ }是讓tmp退出{ }時自動析構*/

5.Vector 記憶體管理成員函式的行為測試

       C++ STL的vector使用非常廣泛,但是對其記憶體的管理模型一直有多種猜測,下面用例項程式碼測試來了解其記憶體管理方式,測試程式碼如下:

#include <iostream>
#include <vector>
using namespace std;

int main()
{
vector<int> iVec;
cout << "容器 大小為: " << iVec.size() << endl;
cout << "容器 容量為: " << iVec.capacity() << endl; //1個元素, 容器容量為1

iVec.push_back(1);
cout << "容器 大小為: " << iVec.size() << endl;
cout << "容器 容量為: " << iVec.capacity() << endl; //2個元素, 容器容量為2

iVec.push_back(2);
cout << "容器 大小為: " << iVec.size() << endl;
cout << "容器 容量為: " << iVec.capacity() << endl; //3個元素, 容器容量為4

iVec.push_back(3);
cout << "容器 大小為: " << iVec.size() << endl;
cout << "容器 容量為: " << iVec.capacity() << endl; //4個元素, 容器容量為4

iVec.push_back(4);
iVec.push_back(5);
cout << "容器 大小為: " << iVec.size() << endl;
cout << "容器 容量為: " << iVec.capacity() << endl; //5個元素, 容器容量為8

iVec.push_back(6);
cout << "容器 大小為: " << iVec.size() << endl;
cout << "容器 容量為: " << iVec.capacity() << endl; //6個元素, 容器容量為8

iVec.push_back(7);
cout << "容器 大小為: " << iVec.size() << endl;
cout << "容器 容量為: " << iVec.capacity() << endl; //7個元素, 容器容量為8

iVec.push_back(8);
cout << "容器 大小為: " << iVec.size() << endl;
cout << "容器 容量為: " << iVec.capacity() << endl; //8個元素, 容器容量為8

iVec.push_back(9);
cout << "容器 大小為: " << iVec.size() << endl;
cout << "容器 容量為: " << iVec.capacity() << endl; //9個元素, 容器容量為16
/* vs2005/8 容量增長不是翻倍的,如 
    9個元素   容量9 
    10個元素 容量13 */

/* 測試effective stl中的特殊的交換 swap() */
cout << "當前vector 的大小為: " << iVec.size() << endl;
cout << "當前vector 的容量為: " << iVec.capacity() << endl;
vector<int>(iVec).swap(iVec);

cout << "臨時的vector<int>物件 的大小為: " << (vector<int>(iVec)).size() << endl;
cout << "臨時的vector<int>物件 的容量為: " << (vector<int>(iVec)).capacity() << endl;
cout << "交換後,當前vector 的大小為: " << iVec.size() << endl;
cout << "交換後,當前vector 的容量為: " << iVec.capacity() << endl;

return 0;
}

6.vector的其他成員函式

        c.assign(beg,end):將[beg; end)區間中的資料賦值給c。
        c.assign(n,elem):將n個elem的拷貝賦值給c。 
        c.at(idx):傳回索引idx所指的資料,如果idx越界,丟擲out_of_range。 
        c.back():傳回最後一個數據,不檢查這個資料是否存在。
        c.front():傳回地一個數據。 
        get_allocator:使用建構函式返回一個拷貝。 
        c.rbegin():傳回一個逆向佇列的第一個資料。 
        c.rend():傳回一個逆向佇列的最後一個數據的下一個位置。 
        c.~ vector <Elem>():銷燬所有資料,釋放記憶體。    

7.備註:在用vector的過程中的一些問題,特此列出討論:

               1)

                    vector <int > a;

                    int  b = 5;

                    a.push_back(b);

                    此時若對b另外賦值時不會影響a[0]的值

                2)

vector <int*> a;
                     int *b;
                     b= new int[4];
                     b[0]=0;
                     b[1]=1;
                     b[2]=2;
                     a.push_back(b);
                     delete b;          //釋放b的地址空間
                     for(int i=0 ; i <3 ; i++)
                     {
                           cout<<a[0][i]<<endl;
                     }

                     此時輸出的值並不是一開始b陣列初始化的值,而是一些無法預計的值.

                    分析:根據1) 2)的結果,可以想到,在1)中,  往a向量中壓入的是b的值,即a[0]=b,此時a[0]和b是儲存在兩個不同的地址中的.因此改變b的值不會影響a[0];而在2)中,因為是把一個地址(指標)壓入向量a,即a[0]=b,因此釋放了b的地址也就釋放了a[0]的地址,因此a[0]陣列中存放的數值也就不得而知了.   

相關推薦

C++ vector用法

vector 是向量型別,它可以容納許多型別的資料,如若干個整數,所以稱其為容器。vector 是C++ STL的一個重要成員,使用它時需要包含標頭檔案: #include<vector>; 一、vector 的初始化:可以有五種方式,舉例說明如下:

c++vector用法

     用法:           1.檔案包含:                 首先在程式開頭處加上#include<vector>以包含所需要的類檔案vector           還有一定要加上using namespace std;           2.變數宣告:      

C++ STL vector 用法

介紹 vector是表示可變大小陣列的序列容器。 就像陣列一樣,vector也採用的連續儲存空間來儲存元素。也就是意味著可以採用下標對vector的元素進行訪問,和陣列一樣高效。但是又不像陣列,它的大小是可以動態改變的,而且它的大小會被容器自動處理。 本質講

C# ListView用法

ont 結束 server 發生 匹配 鼠標 之前 小圖標 order 一、ListView類 1、常用的基本屬性: (1)FullRowSelect:設置是否行選擇模式。(默認為false) 提示:只有在Details視圖該屬性才有意義

C# ListView用法(轉)

分組 創建 cti 排列 checkbox 定義 com 程序 erl 一、ListView類 1、常用的基本屬性: (1)FullRowSelect:設置是否行選擇模式。(默認為false) 提示:只有在Details視圖該屬性才有

C++ string 用法--原作者是Nicolai M.Josuttis

任何人對本文進行引用都要標明作者是Nicolai M.Josuttis /////////////////////////////////////////////////////////////////////////////////// C++ 語言是個十分優秀的語言,但優秀並不表

C/C++ typedef用法

第一、四個用途 用途一: 定義一種型別的別名,而不只是簡單的巨集替換。可以用作同時宣告指標型的多個物件。比如: char* pa, pb; // 這多數不符合我們的意圖,它只聲明瞭一個指向字元變數的指標,  // 和一個字元變數; 以下則可行: typedef char*

C/C++ typedef用法(真的很詳細)

是p2++出錯了。這個問題再一次提醒我們:typedef和#define不同,它不是簡單的文字替換。上述程式碼中const pStr p2並不等於const char * p2。const pStr p2和const long x本質上沒有區別,都是對變數進行只讀限制,只不過此處變數p2的資料型別是我們自

C# ListView用法 很完整

http://blog.csdn.net/xiaohan2826/article/details/8603015 一、ListView類           1、常用的基本屬性:         (1)FullRowSelect:設定是否行選擇模式。(預設為

C#-using用法

    ③可以有多個物件與 using 語句一起使用,但是必須在 using 語句內部宣告這些物件。        using (Font font3=new Font("Arial",10.0f), font4=new Font("Arial",10.0f))       {           // Us

c++ pair 用法

標準庫型別--pair型別定義在utility標頭檔案中定義 1、pair的建立和初始化 pair包含兩個數值,與容器一樣,pair也是一種模板型別。但是又與之前介紹的容器不同,在建立pair物件時,必須提供兩個型別名,兩個對應的型別名的型別不必相同 pair<string,string>a

C++ set 用法

set 是數學中“集合”概念的實現,故 set 中不會存在任意兩個相同的元素。 定義 set<int> set_1 初始化 由於 set 中不允許有重複的元素,故 set 物件只接受列表初始化(C11 標準支援),如 set<int> set

C++ string 用法

任何人對本文進行引用都要標明作者是Nicolai M.Josuttis /////////////////////////////////////////////////////////////////////////////////// C++ 語言是個十分優秀的語言,但

jstl標籤: c:Foreach用法

為迴圈控制,它可以將集合(Collection)中的成員循序瀏覽一遍。運作方式為當條件符合時,就會持續重複執行的本體內容。 語法語法1:迭代一集合物件之所有成員 <c:forEach [var="varName"] items="collection" [varSta

關於c:forEach用法

轉載於:http://blog.csdn.net/honey_claire/article/details/7664165在JSP的開發中,迭代是經常要使用到的操作。例如,逐行的顯示查詢的結果等。在早期的JSP中,通常使用Scriptlets來實現Iterator或者Enum

C++STL之Vector向量,用法和例子 一起學習 一起加油

                                            &

C++ vector用法!!函式,實現)

1,簡述一下vector的基本操作,它的size,capacity(),clear,reverse,reserve,   push_back等!!! 2,說說,vector的儲存特性,是順序儲存還是如同連結串列般,如果是順序儲存的話,那麼是如何執行   erase,inse

C++模版STL中 map 和 string, vector用法

參考: 1. map 用法詳解 std map是STL的一個關聯容器,它提供一對一(其中第一個可以稱為關鍵字,每個關鍵字只能在map中出現一次,第二個可能稱為該關鍵字的值)的資料處理能力,由於這個特性,它完成有可能在我們處理一對一資料的時候,在程式設計上提

[c++] vector中insert()的用法

iterator insert( iterator loc, const TYPE &val ); void insert( iterator loc, size_type num, const TYPE &val ); void insert( itera

C++vector容器用法

C++  vector 是一個十分有用的容器,包含在標頭檔案<vector>中 vector標準庫提供了許多重要的操作 v.empty()  如果v為空,返回true,否則,返回FALSE v.size()    返回v中元素的個數 v.push_back(t)