C++11新特性【下】{lambda表達式、可變模板參數、包裝器}

一、lambda表達式

在C++98中,如果想要對一個數據集合中的元素進行排序,可以使用std::sort方法。如果待排序元素為自定義類型,需要用戶定義排序時的比較規則,隨著C++語法的發展,人們開始覺得上面的寫法太復雜了,每次為了實現一個algorithm算法, 都要重新去寫一個類,如果每次比較的邏輯不一樣,還要去實現多個類,特別是相同類的命名, 這些都給編程者帶來了極大的不便。因此,在C++11語法中出現了Lambda表達式。

1.1 lambda表達式語法:

lambda表達式書寫格式:[capture-list] (parameters) mutable -> return-type { statement }

lambda表達式各部分說明
  • [capture-list] : 捕捉列表,該列表總是出現在lambda函數的開始位置,編譯器根據[ ]來 判斷接下來的代碼是否為lambda函數,捕捉列表能夠捕捉上下文中的變量供lambda 函數使用。
  • (parameters):參數列表。與普通函數的參數列表一致,如果不需要參數傳遞,則可以連同()一起省略
  • mutable:默認情況下,lambda函數總是一個const函數,mutable可以取消其常量性。使用該修飾符時,參數列表不可省略(即使參數為空)。
  • ->returntype:返回值類型。用追蹤返回類型形式聲明函數的返回值類型,沒有返回值時此部分可省略。返回值類型明確情況下,也可省略,由編譯器對返回類型進行推導。
  • {statement}:函數體。在該函數體內,除了可以使用其參數外,還可以使用所有捕獲 到的變量。

注意:

在lambda函數定義中,參數列表和返回值類型都是可選部分,而捕捉列表和函數體可以為空。因此C++11中最簡單的lambda函數為:[]{}; 該lambda函數不能做任何事情。

int main()
{//lambda實現兩數相加auto add = [](int a, int b) {return a + b; };cout << add(1, 2) << endl;//lambda實現swapauto swap = [](int& a, int& b){int temp = a;a = b;b = temp;};
}
捕獲列表說明:

捕捉列表描述了上下文中那些數據可以被lambda使用,以及使用的方式傳值還是傳引用。

  • [var]:表示值傳遞方式捕捉變量var
  • [=]:表示值傳遞方式捕獲所有父作用域中的變量(包括this)
  • [&var]:表示引用傳遞捕捉變量var
  • [&]:表示引用傳遞捕捉所有父作用域中的變量(包括this)
  • [this]:表示值傳遞方式捕捉當前的this指針

捕捉列表還可以混合捕捉:

	int a = 1;int b = 2;int c = 3;int d = 4;//除了a b傳值捕捉外,其他的變量都是引用捕捉auto func = [&, a, b]() mutable{a++;b++;c++;d++;};func();cout << a << " " << b <<" " << c << " " << d << " " << endl;// 1 2 4 5

注意:

  • 父作用域指包含lambda函數的語句塊
  • 捕捉列表不允許變量重復傳遞,否則就會導致編譯錯誤。比如:[=, a]:=已經以值傳遞方式捕捉了所有變量,捕捉a重復
  • 在塊作用域以外的lambda函數捕捉列表必須為空。
  • 在塊作用域中的lambda函數僅能捕捉父作用域中局部變量,捕捉任何非此作用域或者非局部變量都會導致編譯報錯。
  • lambda表達式之間不能相互賦值,即使看起來類型相同

ps:lambda表達式的類型不是直接由語言定義的具名類型,而是編譯器為每一個lambda表達式生成的唯一的、匿名的非聯合類類型(non-union class type)。這個類型沒有名字,因此你不能用它來直接聲明變量或作為函數的參數類型。但是,你可以通過auto關鍵字或者模板來間接地引用這個類型。即使兩個表達式的內容完全相同,其類型都是不一樣的

1.2?函數對象與lambda表達式

函數對象,又稱為仿函數,即可以想函數一樣使用的對象,就是在類中重載了operator()運算符的 類對象。

class Rate
{
public:Rate(double rate) : _rate(rate){}double operator()(double money, int year){return money * _rate * year;}
private:double _rate;
};
int main()
{// 函數對象double rate = 0.49;Rate r1(rate);r1(10000, 2);// lamberauto r2 = [=](double monty, int year)->double {return monty * rate * year;};r2(10000, 2);return 0;
}

從使用方式上來看,函數對象與lambda表達式完全一樣。 函數對象將rate作為其成員變量,在定義對象時給出初始值即可,lambda表達式通過捕獲列表可 以直接將該變量捕獲到。

實際在底層編譯器對于lambda表達式的處理方式,完全就是按照函數對象的方式處理的,即:如 果定義了一個lambda表達式,編譯器會自動生成一個類,在該類中重載了operator()。

二、新的類功能

默認成員函數

原來C++類中,有6個默認成員函數:

  1. 構造函數
  2. 析構函數
  3. 拷貝構造函數
  4. 拷貝賦值重載
  5. 取地址重載
  6. const 取地址重載

最后重要的是前4個,后兩個用處不大。默認成員函數就是我們不寫編譯器會生成一個默認的。 C++11 新增了兩個:移動構造函數和移動賦值運算符重載。

針對移動構造函數和移動賦值運算符重載有一些需要注意的點如下:

  • 如果你沒有自己實現移動構造函數,且沒有實現析構函數 、拷貝構造、拷貝賦值重載中的任 意一個。那么編譯器會自動生成一個默認移動構造。默認生成的移動構造函數,對于內置類 型成員會執行逐成員按字節拷貝,自定義類型成員,則需要看這個成員是否實現移動構造, 如果實現了就調用移動構造,沒有實現就調用拷貝構造。
  • 如果你沒有自己實現移動賦值重載函數,且沒有實現析構函數 、拷貝構造、拷貝賦值重載中 的任意一個,那么編譯器會自動生成一個默認移動賦值。默認生成的移動構造函數,對于內置類型成員會執行逐成員按字節拷貝,自定義類型成員,則需要看這個成員是否實現移動賦 值,如果實現了就調用移動賦值,沒有實現就調用拷貝賦值。(默認移動賦值跟上面移動構造 完全類似)
  • 如果你提供了移動構造或者移動賦值,編譯器不會自動提供拷貝構造和拷貝賦值。
強制生成默認函數的關鍵字default:

C++11可以讓你更好的控制要使用的默認函數。假設你要使用某個默認的函數,但是因為一些原 因這個函數沒有默認生成。比如:我們提供了拷貝構造,就不會生成移動構造了,那么我們可以 使用default關鍵字顯示指定移動構造生成。

class Person
{
public:Person(const char* name = "", int age = 0):_name(name), _age(age){}Person(const Person& p):_name(p._name),_age(p._age){}Person(Person&& p) = default;private:bit::string _name;int _age;
};int main()
{Person s1;Person s2 = s1;Person s3 = std::move(s1);return 0;
}
禁止生成默認函數的關鍵字delete:

如果能想要限制某些默認函數的生成,在C++98中,是J將該函數設置成private,并且只聲明補不定義,這樣只要其他人想要調用就會報錯。在C++11中更簡單,只需在該函數聲明加上=delete即 可,該語法指示編譯器不生成對應函數的默認版本,稱=delete修飾的函數為刪除函數。

class Person
{
public:Person(const char* name = "", int age = 0):_name(name), _age(age){}Person(const Person& p) = delete;
private:bit::string _name;int _age;
};
int main()
{Person s1;Person s2 = s1;Person s3 = std::move(s1);return 0;
}

三、可變模版參數

C++11的新特性可變參數模板能夠讓您創建可以接受可變參數的函數模板和類模板,相比 C++98/03,類模版和函數模版中只能含固定數量的模版參數,可變模版參數無疑是一個巨大的改 進。由于可變模版參數比較抽象難懂,本篇博客主要講解其基本特性和使用方法

下面就是一個基本可變參數的函數模板:

// Args是一個模板參數包,args是一個函數形參參數包
// 聲明一個參數包Args...args,這個參數包中可以包含0到任意個模板參數。
template <class ...Args>
void ShowList(Args... args)
{}

上面的參數args前面有省略號,所以它就是一個可變模版參數,我們把帶省略號的參數稱為“參數 包”,它里面包含了0到N(N>=0)個模版參數。我們無法直接獲取參數包args中的每個參數的, 只能通過展開參數包的方式來獲取參數包中的每個參數,這是使用可變模版參數的一個主要特 點,也是最大的難點,即如何展開可變模版參數。由于語法不支持使用args[i]這樣方式獲取可變參數,所以我們的用一些奇招來一一獲取參數包的值。

遞歸函數方式展開參數包
template<class T>
void _Show_list(const T& val)
{cout << val<<endl;
}template<class T,class ...Args>
void _Show_list(T val,Args... args)
{cout << val << " ";_Show_list(args...);
}template<class ...Args>
void Show_list(Args... args)
{_Show_list(args...);
}int main()
{Show_list(1);Show_list(1,"abc");Show_list(1,"abc",2.2);return 0;
}

底層解釋:

逗號表達式展開參數包

這種展開參數包的方式,不需要通過遞歸終止函數,是直接在expand函數體中展開的, printarg 不是一個遞歸終止函數,只是一個處理參數包中每一個參數的函數。這種就地展開參數包的方式 實現的關鍵是逗號表達式。我們知道逗號表達式會按順序執行逗號前面的表達式。 expand函數中的逗號表達式:(printarg(args), 0),也是按照這個執行順序,先執行 printarg(args),再得到逗號表達式的結果0。同時還用到了C++11的另外一個特性——初始化列 表,通過初始化列表來初始化一個變長數組, {(printarg(args), 0)...}將會展開成((printarg(arg1),0), (printarg(arg2),0), (printarg(arg3),0), etc... ),最終會創建一個元素值都為0的數組int arr[sizeof... (Args)]。由于是逗號表達式,在創建數組的過程中會先執行逗號表達式前面的部分printarg(args) 打印出參數,也就是說在構造int數組的過程中就將參數包展開了,這個數組的目的純粹是為了在 數組構造的過程展開參數包

template<class T>
void print(const T& val)
{cout << val << " ";
}template<class ...Args>
void Show_list(Args... args)
{int arr[] = { (print(args),0)... };cout << endl;
}int main()
{Show_list(1,"abc",2.2);return 0;
}
STL容器中的empalce相關接口函數:
template <class... Args>
void emplace_back (Args&&... args);

可見emplace系列使用了可變模版參數和萬能引用,相比于emplace系列,push_back的參數存在左值引用和右值引用兩個版本,那emplace與push_back有什么區別嗎?

如果插入的數據是一個對象的話,不論是傳左值還是傳右值,兩者的效率是一摸一樣的:

emplace系列函數可以直接傳構造對象的參數包,該參數包會一直往下傳,到最底層時直接構造,相比于push_back傳右值來說省了一次移動構造的消耗

總結:

1.對于深拷貝的類,push_back傳右值和emplace傳構造對象的參數包來說,效率差不多,因為對于深拷貝來說,多一次數據交換的影響不大

2.對于淺拷貝的類來說,push_back傳右值要經歷構造和拷貝構造,emplace傳構造對象的參數包僅一次構造,此時效率提升比較大

3.推薦用emplace系列函數代替push和insert系列函數,此外emplace系列函數能傳參數包就傳參數包

四、包裝器

function包裝器

對于可調用對象一般有三種:函數指針、仿函數、lambda表達式,function包裝器 也叫作適配器。C++中的function本質是一個類模板,也是一個包裝器。包裝器一般就是包裝這三種類型的。

std::function在頭文件<functional>// 類模板原型如下
template <class T> function; ? ? // undefined
template <class Ret, class... Args>
class function<Ret(Args...)>;模板參數說明:
Ret: 被調用函數的返回類型
Args…:被調用函數的形參
包裝器的使用
// 使用方法如下:
#include <functional>
int f(int a, int b)
{return a + b;
}
struct Functor
{
public:int operator() (int a, int b){return a + b;}
};
class Plus
{
public:static int plusi(int a, int b){return a + b;}double plusd(double a, double b){return a + b;}
};
int main()
{// 函數名(函數指針)std::function<int(int, int)> func1 = f;cout << func1(1, 2) << endl;// 函數對象std::function<int(int, int)> func2 = Functor();cout << func2(1, 2) << endl;// lamber表達式std::function<int(int, int)> func3 = [](const int a, const int b){return a + b; };cout << func3(1, 2) << endl;// 類的成員函數std::function<int(int, int)> func4 = &Plus::plusi;cout << func4(1, 2) << endl;std::function<double(Plus, double, double)> func5 = &Plus::plusd;cout << func5(Plus(), 1.1, 2.2) << endl;std::function<double(Plus*, double, double)> func6 = &Plus::plusd;Plus plus;cout << func6(&plus, 1.1, 2.2) << endl;return 0;
}
注意:

包裝器也可以包裝類成員函數,不過要添加函數所屬類域,在c++中一般還要在類名前加一個&,對于靜態成員函數,按照函數聲明傳參即可,對于非靜態成員函數,要注意函數存在一個隱藏的this指針,我們可以傳一個對象,也可以傳一個對象的地址,底層會根據這個對象或者地址從而找到這個函數

包裝器的意義
#include<iostream>
using namespace std;template<class F, class T>
T useF(F f, T x)
{static int count = 0;cout << "count:" << ++count << endl;cout << "count:" << &count << endl;return f(x);
}double f(double i)
{return i / 2;
}struct Functor
{double operator()(double d){return d / 3;}
};int main()
{// 函數名cout << useF(f, 11.11) << endl;cout << endl;// 函數對象cout << useF(Functor(), 11.11) << endl;cout << endl;// lamber表達式cout << useF([](double d)->double { return d / 4; }, 11.11) << endl;cout << endl;return 0;
}

通過上面的程序驗證,我們會發現useF函數模板實例化了三份,如此豐富的可調用類型,可能會導致模板的效率低下。而包裝器就可以很好的解決上述問題。

int main()
{// 函數名std::function<double(double)> func1 = f;cout << useF(func1, 11.11) << endl;cout << endl;// 函數對象std::function<double(double)> func2 = Functor();cout << useF(func2, 11.11) << endl;cout << endl;// lamber表達式std::function<double(double)> func3 = [](double d)->double { return d /4; };cout << useF(func3, 11.11) << endl;cout << endl;return 0;
}

這樣useF就僅實例化出一份,比較節省空間

bind綁定

std::bind函數定義在頭文件中,是一個函數模板,它就像一個函數包裝器(適配器),接受一個可 調用對象(callable object),生成一個新的可調用對象來“適應”原對象的參數列表。一般而 言,我們用它可以把一個原本接收N個參數的函數fn,通過綁定一些參數,返回一個接收M個(M 可以大于N,但這么做沒什么意義)參數的新函數。同時,使用std::bind函數還可以實現參數順 序調整等操作。

// 原型如下:
template <class Fn, class... Args>
/* unspecified */ bind (Fn&& fn, Args&&... args);// with return type (2) 
template <class Ret, class Fn, class... Args>
/* unspecified */ bind (Fn&& fn, Args&&... args);

可以將bind函數看作是一個通用的函數適配器,它接受一個可調用對象,生成一個新的可調用對 象來“適應”原對象的參數列表。 調用bind的一般形式:auto newCallable = bind(callable,arg_list); 其中,newCallable本身是一個可調用對象,arg_list是一個逗號分隔的參數列表,對應給定的 callable的參數。當我們調用newCallable時,newCallable會調用callable,并傳給它arg_list中 的參數。

通過bind改變函數的參數順序

arg_list中的參數可能包含形如_n的名字,其中n是一個整數,這些參數是“占位符”,表示 newCallable的參數,它們占據了傳遞給newCallable的參數的“位置”。數值n表示生成的可調用對 象中參數的位置:_1為newCallable的第一個參數,_2為第二個參數,以此類推。這些占位符存在placehoders的命名空間中

通過bind改變函數的傳參個數
void func(string s, int a, int b)
{cout << s << " " << a << " " << b << endl;
}int main()
{auto f1 = bind(func,"abc" ,placeholders::_1, placeholders::_2);f1(1, 2);auto f2 = bind(func, placeholders::_1,666, placeholders::_2);f2("def", 1);
}結果:
//abc 1 2
//def 666 1

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/web/39345.shtml
繁體地址,請注明出處:http://hk.pswp.cn/web/39345.shtml
英文地址,請注明出處:http://en.pswp.cn/web/39345.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

深入理解Git:checkout的本質與原理深度解析

在Git的版本控制世界中&#xff0c;checkout是一個極其重要且功能豐富的命令&#xff0c;它貫穿于日常開發的各個環節。然而&#xff0c;隨著Git版本的更新迭代&#xff0c;checkout的功能逐漸被新的命令如switch和restore所分解&#xff0c;但這并不妨礙我們深入理解其本質與原…

#練習 #綜合 LinuxC高級

一、選擇題 1.常見的Linux發行版本有很多&#xff0c;下面不是Linux發行版本的是( ) A. Red Hat Linux B. Ubuntu Linux C.X Window D.SuSE Linux 答案 2.下面不是對Linux操作系統特點描述的是( ) A.良好的可移植性 B.單用戶 C.多用戶 D.多任務 答案…

1-訊飛星火大模型API調用示例解析

1官網鏈接 比賽官網&#xff1a;2024 iFLYTEK A.I.開發者大賽-訊飛開放平臺 (xfyun.cn)&#xff1b;控制臺官網&#xff1a;控制臺-訊飛開放平臺 (xfyun.cn)&#xff1b; 2星火模型python調用示例 示例鏈接&#xff1a;https://xfyun-doc.xfyun.cn/lc-sp-PythonDemo-17163704…

了解MySQL【事務】的功能:確保數據完整性的關鍵技術

在數據庫管理中&#xff0c;事務是確保數據完整性和一致性的核心機制。特別是對MySQL這樣廣泛應用的開源數據庫系統&#xff0c;掌握事務的使用至關重要。在這篇文章中&#xff0c;我們將全面探討MySQL事務的工作原理、ACID屬性、隔離級別以及最佳實踐&#xff0c;從而幫助開發…

寶塔Linux面板配置環境 + 創建站點

一、安裝 &#xff08;1&#xff09;進入寶塔官網 https://www.bt.cn/new/index.html &#xff08;2&#xff09;點擊“ 立即免費安裝 ”&#xff0c;選擇 Centos安裝腳本 &#xff08;3&#xff09;進入 ssh 輸入以下命令安裝寶塔 yum install -y wget && wget -O …

實驗三 SQL Server SSMS工具添加數據

1、打開ecommerce數據庫&#xff0c;向表中分別錄入以下數據信息 &#xff08;1&#xff09;商品類別表category catno catname describe 101 手機 各種品牌、型號手機 201 激光打印機 各種激光打印機 202 噴墨打印機 各種噴墨打印機 301 平板電腦 各種平板電腦…

Python后端面試題

1. 文件操作w和r的區別 在Python中&#xff0c;文件操作模式中的w和r都表示對文件的讀寫操作&#xff0c;但它們在打開文件時的行為有所不同&#xff1a; r模式&#xff1a; 讀寫&#xff1a;這種模式允許你同時讀取和寫入文件。文件必須已經存在&#xff0c;否則會拋出一個Fi…

思看科技募資額驟降:對賭壓力下巨額分紅,還購買 7項商業房產

《港灣商業觀察》施子夫 6月11日&#xff0c;證監會網站披露思看科技&#xff08;杭州&#xff09;股份有限公司&#xff08;以下簡稱&#xff0c;思看科技&#xff09;的首輪審核問詢函回復意見并更新2023年財務數據&#xff0c;繼續推進上市進程。 公開信息顯示&#xff0c…

低空經濟背景下的無人機人才培養

無人機作為低空經濟的重要組成部分&#xff0c;其技術的進步和應用的拓展直接推動了低空經濟的快速發展。無人機以其高效、靈活、低成本的特點&#xff0c;在多個領域發揮了重要作用&#xff0c;成為推動低空經濟發展的重要力量。 無人機人才培養的重要性 隨著無人機應用的不…

深度之眼(二十九)——神經網絡基礎知識(四)-循環神經網絡

文章目錄 一、 學習目標二、序列數據三、語言模型四、循環神經網絡4.1 RNN的反向傳播 五、門控循環單元-GNU5.1 候選隱藏狀態 六、長短期記憶網絡-LSTM七、回顧 一、 學習目標 二、序列數據 序列數據是常見的數據類型&#xff0c;前后數據通常具有關聯性 三、語言模型 綜合…

【技術雜談】如何訪問Github | 解決無法連接Github的問題

訪問網頁的過程 什么是域名&#xff1f;什么是IP地址&#xff1f;- 域名是網站的名稱。 - IP地址是服務器在互聯網上的邏輯地址。域名往往是固定的&#xff0c;但是IP地址很有可能是會改變的。計算機通過Host文件檢查本地緩存是否有域名對應IP地址 Host文件路徑 C:\Windows\Sy…

C#反射基本應用

1、反射 反射是.NET Framework的一個特性&#xff0c;它允許在運行時獲取類型的信息以及動態創建對象&#xff0c;調用方法&#xff0c;以及訪問字段和屬性。 2、代碼 using System; using System.Collections.Generic; using System.Linq; using System.Reflection; using Sy…

Node.js path模塊

在 Node.js 中&#xff0c;path 模塊用于處理和轉換文件路徑。以下是一些常用的 path 模塊方法及其說明&#xff1a; path.basename(path[, ext]) 返回路徑中的最后一部分&#xff0c;即文件名。示例&#xff1a;const path require(path); console.log(path.basename(/foo/ba…

ShardingSphere分庫分表+讀寫分離

ShardingSphere 是一個開源的分布式數據庫中間件&#xff0c;它支持分庫分表和讀寫分離的功能&#xff0c;可以有效地提高數據庫的并發處理能力和數據存儲能力。以下是關于 ShardingSphere 分庫分表和讀寫分離的一些關鍵點&#xff1a; 1. **讀寫分離**&#xff1a;在 Shardin…

Python中使用Oracle向量數據庫實現文本檢索系統

Python中使用Oracle向量數據庫實現文本檢索系統 代碼分析 在本文中,我們將深入分析一個使用Oracle向量數據庫實現文本檢索系統的Python代碼,并基于相同的技術生成一個新的示例。這個系統允許我們存儲文檔及其嵌入向量,并執行相似性搜索。 代碼分析 讓我們逐步分析原始代碼的主…

探究Executors創建的線程池(如newFixedThreadPool)其核心線程數等參數的可調整性

java中提供Executors類來創建一些固定模板參數的線程池&#xff0c;如下圖&#xff08;newWorkStealingPool除外&#xff0c;這個是創建ForkJoinPool的&#xff0c;這里忽略&#xff09;&#xff1a; 拿newFixedThreadPool方法創建線程池為例&#xff0c;newFixedThreadPool是…

白楊SEO:打粉是啥?打粉引流怎么做?打粉引流犯法嗎?小紅書代發效果好嗎?

文章大綱&#xff1a; 1、打粉是什么意思&#xff1f; 2、打粉有哪些方法&#xff1f; 3、打粉一般怎么變現&#xff1f; 4、打粉引流是違法犯罪嗎&#xff1f; 5、小紅書代發是啥&#xff1f; 6、小紅書批量代發效果好嗎&#xff1f; 打粉是什么意思&#xff1f; 打粉這…

第1章 firewalld防火墻

防火墻 概念 支持網絡區域所定義的網絡鏈接以及接口安全等級的動態防火墻管理工具支持IPv4、IPv6防火墻設置以及以太網橋支持服務或應用程序直接添加防火墻規則接口擁有兩種配置模式 運行時配置&#xff1a;添加的策略立即生效&#xff0c;不用重載防火墻&#xff0c;策略臨時…

C語言-初探指針

初探指針 指針概念指針和指針類型指針類型意義 野指針如何避免 指針運算指針-整數指針-指針指針的關系運算 指針和數組二級指針指針數組 指針概念 指針是內存中一個最小單元(1個字節)的編號&#xff0c;也就是地址平時口語中說的指針&#xff0c;通常指的是指針變量&#xff0…

(免費領源碼)java#springboot#mysql校園醫院預約掛號系統32236-計算機畢業設計項目選題推薦

摘 要 信息化社會內需要與之針對性的信息獲取途徑&#xff0c;但是途徑的擴展基本上為人們所努力的方向&#xff0c;由于站在的角度存在偏差&#xff0c;人們經常能夠獲得不同類型信息&#xff0c;這也是技術最為難以攻克的課題。針對校園醫院管理等問題&#xff0c;對校園醫院…