欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

C++STL教程之vector模板的使用

 更新時間:2022年08月22日 08:43:35   作者:Shawn-Summer  
模板類vector是一個動態(tài)數(shù)組,類似于string類,可存放任意類型,能夠末尾、中間增加數(shù)據(jù),基本上是是new創(chuàng)建動態(tài)數(shù)組的替代品,vector可以自動完成new和delete

vector模板類

STL(標準模板庫)提供了容器、迭代器、函數(shù)對象、算法的模板。容器是類似于數(shù)組的東西,它可以存儲若干值,STL容器是同質(zhì)的,即存儲的值的類型相同;迭代器是用來遍歷容器的,它和能遍歷數(shù)組的指針類似,是廣義指針;函數(shù)對象是類似于函數(shù)的對象,可以是類對象和函數(shù)指針;算法就是一些能完成特定任務的處方。

我們來看最簡單的容器:vector模板類。

1. vector模板類

1.1 創(chuàng)建模板類

在頭文件vector中定義了vector模板,我們稱之為矢量,它就像是加強版的數(shù)組。

創(chuàng)建vector模板對象:

std::vector<int> first;                                // empty vector of ints
std::vector<int> myadd(5);                             //vector of five ints
std::vector<int> second (4,100);                       // four ints with value 100
std::vector<int> third (second.begin(),second.end());  // iterating through second
std::vector<int> fourth (third);                       // a copy of third

一般來說,我們使用前三種方式初始化vector對象,創(chuàng)建好了后數(shù)組可以做的它都可以,例如我們可以使用[]隨機訪問數(shù)據(jù)。

矢量模板類還支持列表初始化語句

std::vector<int> a{1,2,3,4,5};

1.2 STL容器都提供的成員方法

  • size():返回容器中元素數(shù)目
  • swap():交換兩個容器的內(nèi)容
  • begin():返回一個指向容器第一個元素的迭代器
  • end():返回一個表示超過容器尾的迭代器

什么是迭代器?它是一個廣義指針。它可以是指針,也可以是一個可對其執(zhí)行–解除引用operator*()和遞增operator++()–的對象。每一個容器類都定義了一個合適的迭代器,它的類型是一個名為iteratortypedef,其作用域是整個類。

我們可以這樣聲明一個迭代器:
vector<double>::iterator pd;
也可以使用auto關鍵字:
auto pd=scores.begin();
我們可以使用迭代器pd進行如下操作:

pd=scores.begin();
*pd=22.3;
++pd;
pd++;
--pd;
pd--;

總之迭代器就相當于是指向容器中元素的指針。

什么是超過結(jié)尾(past-the-end)?它是一種迭代器,指向容器中最后一個元素后面那個元素。例如在C風格字符串中,字符串的末尾的\0就是超過結(jié)尾指向的元素。end()成員函數(shù)會返回超過結(jié)尾迭代器。
那么我們的遍歷可以這樣寫:

for(pd=scores.begin();pd!=scores.end(),pd++)
    cout<<*pd;
#include<vector>
#include<iostream>
int main()
{
    using std::cout;
    using std::endl;
    using std::vector;
    vector<double> a{1,2,3,4,5};
    vector<double> b{6,7,8};
    cout<<"a size: "<<a.size()<<endl;
    cout<<"b size: "<<b.size()<<endl;
    cout<<"a :";
    for(vector<double>::iterator i=a.begin();i!=a.end();i++)
        cout<<*i<<" ";
    cout<<"\nb :";
    for(vector<double>::iterator i=b.begin();i!=b.end();i++)
        cout<<*i<<" ";
    a.swap(b);
    cout<<"\nafter swap:"<<endl;
    cout<<"a :";
    for(vector<double>::iterator i=a.begin();i!=a.end();i++)
        cout<<*i<<" ";
    cout<<"\nb :";
    for(vector<double>::iterator i=b.begin();i!=b.end();i++)
        cout<<*i<<" ";
}

a size: 5
b size: 3
a :1 2 3 4 5
b :6 7 8
after swap:
a :6 7 8
b :1 2 3 4 5

以上代碼是測試了,矢量類的一些接口

實際上還有很多接口:例如empty,front,back;可以直接看cplusplus

1.3 vector特有的成員方法

  • push_back():將元素添加到矢量末尾,而且矢量長度會自動增大
  • erase():刪除給定區(qū)間內(nèi)的元素
  • insert():插入指定區(qū)間內(nèi)的元素

push_back()接受一個元素類型的參數(shù),它相當于在矢量的超過末尾的地方加個元素:

vector<double> scores;
double temp=1.23;
scores.push_back(temp);

erase()接受兩個迭代器參數(shù),這兩個迭代器定義了要刪除的區(qū)間,第一個迭代器是區(qū)間起始處,第二個迭代器是區(qū)間終止后的第一個位置,例如a.erase(start,end);是指刪除區(qū)間[start,end)左開右閉中的元素,而C++中所說的區(qū)間都是這種左開右閉的區(qū)間。

scores.erase(scores.begin(),scores.begin()+2);

上面這句代碼就會刪除矢量對象中前兩個元素。

insert()會把指定區(qū)間里的元素插到一個位置前面。它接受三個迭代器參數(shù),第一個參數(shù)指出新元素的插入位置,第二第三就是區(qū)間;

vector<int> old_v;
vector<int> new_v;
...
old_v.insert(old_v.begin(),new_v.begin()+1,new_v.end());

上面這段代碼會把new_v中除了第一個元素外的所有元素插到old_v的第一個元素的前面。

超尾元素的存在,使得在最后一個元素后面插入元素變得簡單:

old_v.insert(old_v.end(),new_v.begin()+1,new_v.end());

#include<vector>
#include<iostream>
int main()
{
    using namespace std;
    vector<int> a;
    a.push_back(1);
    a.push_back(2);
    cout<<"a: ";
    for(auto i=a.begin();i!=a.end();i++)
    cout<<*i;
    cout<<endl;
    vector<int>b{3,4,5,6,7};
    a.insert(a.end(),b.begin(),b.begin()+3);
    cout<<"after insert: ";
    cout<<"a: ";
    for(auto i=a.begin();i!=a.end();i++)
    cout<<*i;
    cout<<endl;
    a.erase(a.begin()+1,a.begin()+3);
    cout<<"after erase: ";
    cout<<"a: ";
    for(auto i=a.begin();i!=a.end();i++)
    cout<<*i;
    cout<<endl;
}

a: 12
after insert: a: 12345
after erase: a: 145   

1.4 STL容器的非成員方法

我們會對容器做很多操作,例如搜索,排序。但是這些功能不會放在成員方法中,因為不同的容器類的排序或者搜索方法都是類似的,所以我們?yōu)榱斯?jié)省代碼,就不會為每個容器單獨寫這種成員方法。但是,即使存在執(zhí)行相同任務的非成員函數(shù),STL容器可能也會定義相同的成員方法,例如vectorswap()成員方法比swap()非成員方法效率高,但是非成員函數(shù)讓您可以交換兩個不同類型的容器的內(nèi)容。

  • for_each():遍歷
  • random_shuffle():隨機排列
  • sort():排序

這些方法都定義在頭文件algorithm中,這就是我們所說的算法。
for_each()接受3個參數(shù),前兩個是定義區(qū)間的迭代器,最后一個是指向函數(shù)的指針(或者說是函數(shù)對象)。for_each()將被指向的函數(shù)應用于容器間的各個元素。但是for_each()不能修改容器的元素值。我們可以使用它來代替for循環(huán)。

for_each(books.begin(),books.end(),foo);//foo是函數(shù)名,即函數(shù)地址

這個語法很熟悉,很像基于范圍的for循環(huán):

double prices[5]={4.99,10.99,6.87,7.99,8.49};
for(double x:prices)
    cout<<x<<endl;
for_each(books.begin(),books.end(),foo);//foo是函數(shù)名,即函數(shù)地址
//等價于
for(auto x:books) foo(x);

但是基于范圍的for循環(huán)可以改變?nèi)萜鞯膬?nèi)容,我們只需要函數(shù)的參數(shù)是引用參數(shù)foo(int &);
然后我們的代碼:

for(auto &x:books) foo(x);

random_shuffle()接受兩個指定區(qū)間的迭代器,并隨機排列區(qū)間中的元素,但是random_shuffle()要求容器允許隨機訪問(即使用books[i]可以直接訪問元素)

random_shuffle(books.begin(),books.end());

sort()也要求容器支持隨機訪問。
第一個版本的sort()接受兩個指定區(qū)間的迭代器,并且使用<運算符對容器中的元素進行升序排列:

vector<int> coolstuff;
...
sort(coolstuff.begin(),coolstuff.end());

這就意味著,如果容器中的元素的類型必須定義operator<()。

第二個版本的sort()接受三個參數(shù),它更實用,前兩個參數(shù)是指定區(qū)間的迭代器,第三個參數(shù)是函數(shù)指針(或函數(shù)對象)。這個函數(shù)指針指向一個返回bool值,接受兩個元素的函數(shù),如果true就說明排序正確,如果false就說明排序錯誤。

例如我們希望采用降序排列:

bool compare(double db1,double db2)
{
    if(db1<db2)
        return false;
    else 
        return true;
}
int main(){
vector<double> a{4.99,10.99,6.87,7.99,8.49};
sort(a.begin(),a.end(),compare);
}

或者直接使用函數(shù)對象:

sort(a.begin(),a.end(),greater<double>());

這里使用的greater<double>()就是函數(shù)對象,它返回了double類型的大于運算。

#include<vector>
#include<iostream>
#include<algorithm>
void show(const int &a)
{
    std::cout<<a<<" ";
}
bool greater(const int &x,const int &y)
{
    return x>y;
}
int main()
{
    using std::vector;
    using std::random_shuffle;
    using std::sort;
    using std::for_each;
    using std::cout;
    vector<int> a;
    for(int i=0;i<20;i++)
        a.push_back(i);
    cout<<"initial: ";
    for_each(a.begin(),a.end(),show);
    random_shuffle(a.begin(),a.end());
    cout<<"\nafter shaking: ";
    for_each(a.begin(),a.end(),show);
    sort(a.begin(),a.end());
    cout<<"\nAscending: ";
    for_each(a.begin(),a.end(),show);
    sort(a.begin(),a.end(),greater);
    cout<<"\nDescending: ";
    for_each(a.begin(),a.end(),show);
}

initial: 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19       
after shaking: 12 1 9 2 0 11 7 19 4 15 18 5 14 13 10 16 6 3 8 17 
Ascending: 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19     
Descending: 19 18 17 16 15 14 13 12 11 10 9 8 7 6 5 4 3 2 1 0  

到此這篇關于C++STL教程之vector模板的使用的文章就介紹到這了,更多相關C++ vector模板內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

  • 詳解C++編程中對二進制文件的讀寫操作

    詳解C++編程中對二進制文件的讀寫操作

    這篇文章主要介紹了C++編程中對二進制文件的讀寫操作,是C++入門學習中的基礎知識,需要的朋友可以參考下
    2015-09-09
  • c++支持coroutine的簡單示例

    c++支持coroutine的簡單示例

    這篇文章主要介紹了c++支持coroutine的簡單示例,使用的是linux 平臺做的,需要的朋友可以參考下
    2014-03-03
  • C++函數(shù)重載詳解及實例代碼

    C++函數(shù)重載詳解及實例代碼

    這篇文章主要介紹了C++函數(shù)重載詳解及實例代碼的相關資料,需要的朋友可以參考下
    2016-09-09
  • Windows上使用vs編譯fdk-aac的教程詳解

    Windows上使用vs編譯fdk-aac的教程詳解

    有時需要在Windows上用到fdk-aac,fdk-aac是源碼是cmake的,可以直接轉(zhuǎn)成vs項目,下面就跟隨小編一起學習一下如何在Windows上使用vs編譯fdk-aac吧
    2023-11-11
  • C++之默認參數(shù)詳解

    C++之默認參數(shù)詳解

    這篇文章主要介紹了C++的默認參數(shù),是C++入門學習中的基礎知識,需要的朋友可以參考下,希望能夠給你帶來幫助
    2021-11-11
  • 探討編寫int strlen(char *strDest);不允許定義變量的問題

    探討編寫int strlen(char *strDest);不允許定義變量的問題

    本篇文章是對編寫int strlen(char *strDest);不允許定義變量的問題進行了詳細的分析介紹,需要的朋友參考下
    2013-05-05
  • C++定時器實現(xiàn)和時間輪介紹

    C++定時器實現(xiàn)和時間輪介紹

    這篇文章主要介紹了C++定時器實現(xiàn)和時間輪介紹,定時器可以由很多種數(shù)據(jù)結(jié)構實現(xiàn),比如最小堆、紅黑樹、跳表、甚至數(shù)組都可以,其本質(zhì)都是拿到最小時間的任務,然后取出該任務并執(zhí)行,更多相關內(nèi)容介紹,需要的小伙伴可以參考一下
    2022-09-09
  • 關于C++中數(shù)據(jù)16進制輸出的方法

    關于C++中數(shù)據(jù)16進制輸出的方法

    本文主要介紹了關于C++中數(shù)據(jù)16進制輸出的方法,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2023-03-03
  • C語言數(shù)據(jù)結(jié)構之單向鏈表詳解

    C語言數(shù)據(jù)結(jié)構之單向鏈表詳解

    單向鏈表(單鏈表)是鏈表的一種,其特點是鏈表的鏈接方向是單向的,對鏈表的訪問要通過順序讀取從頭部開始。本文將為大家詳細講講單向鏈表的實現(xiàn)與使用,需要的可以參考一下
    2022-08-08
  • 使用QPainter畫一個3D正方體

    使用QPainter畫一個3D正方體

    這篇文章主要為大家詳細介紹了使用QPainter畫一個3D正方體,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2020-08-08

最新評論