C++ primer超詳細(xì)講解泛型算法
初識(shí)泛型算法
只讀算法
只讀取輸入范圍內(nèi)的函數(shù),不改變?cè)?,find,accumula也是如此
(1)accumulate算法為求和算法,前兩個(gè)參數(shù)指出求和元素范圍,第三個(gè)是和的初值,例:
int sum=accumulate(v.begin(),v.end(),0)
(2)操作兩個(gè)序列的算法
equal算法,確定兩個(gè)序列是否保存相同的值,將第一個(gè)序列的每個(gè)元素和第二個(gè)序列中的每個(gè)元素進(jìn)行比較,若相同返回true,否則返回false,接受三個(gè)參數(shù),前兩個(gè)表示第一個(gè)序列的元素范圍,第三個(gè)表示第二個(gè)序列的首元素
equal(r1.begin(),r1.end(),r2.begin())
寫容器算法
(1)拷貝算法
向另一個(gè)目的位置迭代器指向的輸出序列中的元素寫入數(shù)據(jù)算法。此算法接受三個(gè)迭代器,前兩個(gè)表示一個(gè)舒服范圍,第三個(gè)表示目的序列的起始位置。copy返回目的迭代器的值。
int a1[] = { 0,1,2,3,4,5 }; int a2[sizeof(a1) / sizeof(*a1)]; auto ret = copy(begin(a1), end(a1), a2);
定制操作
lambda表達(dá)式
(1)定義
一個(gè)lambda表達(dá)式表示一個(gè)可調(diào)用的代碼單元,可理解為未命名的內(nèi)聯(lián)函數(shù)
lambda表達(dá)式形式:
[capture list](parameter list) - > return type{function body}
可以忽略參數(shù)列表和返回類型,但必須永遠(yuǎn)包含捕獲列表和函數(shù)體
auto f =[] {return 42}
調(diào)用: cout<<f()<<endl;
(2)向lambda傳遞參數(shù)
實(shí)參被用來(lái)初始化lambda的形參,lambda不能有默認(rèn)參數(shù)
例:
//[](const string &a,const string &b){return a.size()<b.size()} //調(diào)用 sort(w.begin(),w.end(),[](const string &a,const string &b){return a.size()<b.size()})
(3)使用捕獲列表
一個(gè)lambda通過(guò)將局部變量包含在其捕獲列表中指出將會(huì)使用這些變量,捕獲列表指引lambda在其內(nèi)部包含訪問(wèn)局部變量所需的全部信息
例如,找出第一個(gè)大于等于給定長(zhǎng)度的單詞。
函數(shù)biggies實(shí)現(xiàn)
void biggies(vector<string> &words,vector<string>::size_type sz) { auto wc=find_if(words.begin(),words.end(), [sz](const string &a){return a.size()>=sz}); }
lambda捕獲和返回
(1)值捕獲
與傳值參數(shù)類似,采用值捕獲的前提是變量可以拷貝,與參數(shù)不同,被捕獲的變量的值是在lambda創(chuàng)建時(shí)拷貝,而不是調(diào)用時(shí)拷貝
void fcun() { size_t v1=42; auto f=[v1]{return v1;}; v1=0; auto j=f(); }
由于被捕獲變量的值是在lambda創(chuàng)建時(shí)拷貝,因此隨后對(duì)其修改不會(huì)影響到lambda內(nèi)對(duì)應(yīng)的值,上述中j的值為42
(2)引用捕獲
void fcun() { size_t v1=42; auto f=[&v1]{return v1;}; v1=0; auto j=f(); //j為0,f2保存v1的引用,而非拷貝 }
(3)隱式捕獲
可以讓編譯器根據(jù)lambda體中的代碼推斷我們要使用哪些變量,此時(shí)應(yīng)在捕獲列表中寫一個(gè)&或=,&表示采用捕獲引用,=表示采用值捕獲方式。
wc=find_if(w.begin(),w.end(),[=](const string &s){return s.size>=sz;});
(4)可變lambda
在值拷貝的情況下,lambda不會(huì)改變其值,若希望能夠改變被捕獲的變量的值,則在參數(shù)列表首加上mutable關(guān)鍵字。
void func() { size_t v1=42; auto f=[vi]() mutable{return ++v1;}; v1=0; auto j=f(); }
一個(gè)引用捕獲的變量能否可以修改依賴于此引用指向的是一個(gè)const還是非const類型
void fc() { const size_t v1 = 42; auto f = [&v1]() mutable {return ++v1; };//無(wú)法修改v1 auto j = f(); cout << j << endl; }
(5)指定lambda的返回類型
默認(rèn)情況下,如果一個(gè)lambda體中包含return之外的任何語(yǔ)句,則編譯器假定此lambda返回void。
例如:將一個(gè)序列中負(fù)數(shù)替換為其絕對(duì)值
transform(v.begin(),v.end(),v.begin(),[](int i){return i<0?-i:i;})
transform接受三個(gè)參數(shù),前兩個(gè)表示迭代器輸入序列,第三個(gè)表示迭代器目的位置。
上述中,我們無(wú)需指定返回類型,因此可以根據(jù)條件運(yùn)算符的類型推斷出來(lái)。
若改寫成存在if語(yǔ)句,則存在錯(cuò)誤
transform(v.begin(),v.end(),v.begin(),[](int i){if(i<0) return -i; else return i;})
編譯器發(fā)現(xiàn)存在return之外的語(yǔ)句,所以推斷其返回類型為void,而他卻返回了一個(gè)int
正確寫法:
transform(v.begin(),v.end(),v.begin(), [](int i) ->int {if(i<0) return -i; else return i;})
再探迭代器
標(biāo)準(zhǔn)庫(kù)頭文件iterator中還定義了額外幾種迭代器,包括如下:
- 插入迭代器:綁定到一個(gè)容器上,向容器插入元素
- 流迭代器:綁定到輸入輸出流上,可以用來(lái)遍歷所關(guān)聯(lián)的IO流
- 反向迭代器:這些迭代器向后而不是向前移動(dòng),forwar_list不存在此迭代器
- 移動(dòng)迭代器:移動(dòng)元素專用
插入迭代器
例:
list<int> lst = { 1,2,3,4 }; list<int> lst2, lst3; copy(lst.begin(), lst.end(), front_inserter(lst2)); copy(lst.begin(), lst.end(), inserter(lst3, lst3.begin()));
front_inserter總是插入容器的第一個(gè)元素之前,所以lst2中的元素順序?yàn)?,3,2,1,而lst3中的元素順序?yàn)?,2,3,4
iostream迭代器
istream_iterator讀取輸入流,ostream_iterator向一個(gè)輸出流寫數(shù)據(jù)。
(1)istream_iterator操作
創(chuàng)建流迭代器時(shí),必須指定迭代器將要讀寫的對(duì)象類型,可以創(chuàng)建空的初始化迭代器,用作尾后值使用的迭代器,一旦關(guān)聯(lián)的流遇到文件尾或者IO錯(cuò)誤,迭代器的值就與尾后迭代器相等。
例:讀取輸入數(shù)據(jù)保存到vector中
vector<int> v; istream_iterator<int> it(cin); istream_iterator<int> e; while (it != e) v.push_back(*it++); //或 v.push_back(it,e)
(2)使用算法操作流迭代器
例:
istream_iterator<int> it(cin); istream_iterator<int> e; cout << accumulate(it, e, 0) << endl;
(3)ostream_iterator操作
創(chuàng)建一個(gè)ostream_iterator時(shí),可以提供第二參數(shù),他是一個(gè)字符串,在輸出每個(gè)元素后都會(huì)打印此字符串,不允許空的或表示尾后的ostream_iterator
vector<int> v = { 1,2,3,4 }; ostream_iterator<int> out(cout, "@@"); for (auto e : v) *out++ = e; //賦值語(yǔ)句實(shí)際上是將元素寫到cout,且*和++并不做任何事 cout << endl;
反向迭代器
反向迭代器就是在容器中尾元素向首元素反向移動(dòng)的迭代器,遞增一個(gè)反向迭代器會(huì)向前移動(dòng)一個(gè)元素,遞減一個(gè)迭代器會(huì)向后移動(dòng)一個(gè)元素
將反向迭代器轉(zhuǎn)換為普通迭代器可使用reverse_iterator中的base成員函數(shù)來(lái)完成轉(zhuǎn)換
例:輸出最后一個(gè)單詞
string line = { "first,middle,last" }; auto r = find(line.rbegin(), line.rend(), ','); cout << string(r.base(), line.end()) << endl;
到此這篇關(guān)于C++ primer超詳細(xì)講解泛型算法的文章就介紹到這了,更多相關(guān)C++ primer泛型算法內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
C++ 流插入和流提取運(yùn)算符的重載的實(shí)現(xiàn)
這篇文章主要介紹了C++ 流插入和流提取運(yùn)算符的重載的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-12-12基于C語(yǔ)言實(shí)現(xiàn)見(jiàn)縫插針游戲的示例代碼
見(jiàn)縫插針游戲就是使用鼠標(biāo)左鍵點(diǎn)擊發(fā)射針,當(dāng)兩個(gè)針的夾角小于一定限制時(shí),游戲結(jié)束。本文將用C語(yǔ)言實(shí)現(xiàn)這一有趣游戲,感興趣的可以了解一下2022-11-11歸并排序的遞歸實(shí)現(xiàn)與非遞歸實(shí)現(xiàn)代碼
以下是對(duì)歸并排序的遞歸實(shí)現(xiàn)與非遞歸實(shí)現(xiàn)代碼進(jìn)行了詳細(xì)的介紹,需要的朋友可以過(guò)來(lái)參考下2013-08-08