網站首頁 編程語言 正文
lambda 源自希臘字母表中第 11 位的 λ,在計算機科學領域,它則被用來表示一種匿名函數。所謂匿名函數,簡單地理解就是沒有名稱的函數,又常被稱為 lambda 函數或者 lambda 表達式。
1. lambda匿名函數的定義
[capture](parameters)mutable ->return-type{statement}
參數說明:
- [capture]:捕捉列表,[] 是lambda引出符,編譯器根據該引出符判斷接下來的代碼是否是lambda函數。捕捉列表用于捕捉父域中的變量以供lambda函數使用,捕捉列表可以由多個項組成,用","分割。[var]表示以值傳遞方式捕捉父域中的變量var,[=]表示以值傳遞方式捕捉父域中的所有變量(包括this),[&var]表示以引用傳遞方式捕捉父域中的變量var,[&]表示以引用傳遞方式捕捉父域中的所有變量(包括this),[this]表示以值傳遞方式捕捉當前的this指針。
- (parameters):參數列表,與普通函數的參數列表一致,如果不需要參數傳遞,則可以連同括號()一起省略。
- mutable:mutable修飾符,默認情況下,lambda函數總是一個const函數,mutable可以取消其常量性。在使用該修飾符時,參數列表不可省略(即使參數為空)。
- ->return-type:返回類型,用追蹤返回類型形式聲明函數的返回類型,不需要返回值的時候可以連同符號->一起省略。在返回類型明確的情況下,也可以省略該部分,讓編譯器對返回類型進行推導。
- {statement}:函數體,內容與普通函數一樣,不過除了可以使用參數之外,還可以使用所有捕獲的變量。
lambda匿名函數中的[外部變量]
外部變量格式 | 功能 |
---|---|
[] | 空方括號表示當前 lambda 匿名函數中不導入任何外部變量。 |
[=] | 只有一個 = 等號,表示以值傳遞的方式導入所有外部變量; |
[&] | 只有一個 & 符號,表示以引用傳遞的方式導入所有外部變量; |
[val1,val2,...] | 表示以值傳遞的方式導入 val1、val2 等指定的外部變量,同時多個變量之間沒有先后次序; |
[&val1,&val2,...] | 表示以引用傳遞的方式導入 val1、val2等指定的外部變量,多個變量之間沒有前后次序; |
[val,&val2,...] | 以上 2 種方式還可以混合使用,變量之間沒有前后次序。 |
[=,&val1,...] | 表示除 val1 以引用傳遞的方式導入外,其它外部變量都以值傳遞的方式導入。 |
[this] | 表示以值傳遞的方式導入當前的 this 指針。 |
注意,單個外部變量不允許以相同的傳遞方式導入多次。例如 [=,val1] 中,val1 先后被以值傳遞的方式導入了 2 次,這是非法的。
最簡單的lambda匿名函數
[]{}
此 lambda 匿名函數未引入任何外部變量([] 內為空),也沒有傳遞任何參數,沒有指定 mutable、noexcept 等關鍵字,沒有返回值和函數體。所以,這是一個沒有任何功能的 lambda 匿名函數。
2. lambda匿名函數的使用
2.1 lambda匿名函數的定義和使用
#include <iostream>
#include <algorithm>
using namespace std;
int main()
{
int num[4] = { 4, 2, 3, 1 };
// 對數組 num 中的元素進行升序排序
sort(num, num + 4, [=](int x, int y) -> bool { return x < y; });
for (int n : num) {
cout << n << " ";
}
return 0;
}
以上程序通過調用 sort() 函數實現了對 num 數組中元素的升序排序,其中就用到了 lambda 匿名函數。而如果使用普通函數,需以如下代碼實現:
#include <iostream>
#include <algorithm>
using namespace std;
// 自定義的升序排序規則
bool sort_up(int x, int y) {
return x < y;
}
int main()
{
int num[4] = { 4, 2, 3, 1 };
// 對數組 num 中的元素進行升序排序
sort(num, num + 4, sort_up);
for (int n : num) {
cout << n << " ";
}
return 0;
}
此程序中 sort_up() 函數的功能和上一個程序中的 lambda 匿名函數完全相同。顯然在類似的場景中,使用 lambda 匿名函數更有優勢。
除此之外,雖然 lambda 匿名函數沒有函數名稱,但我們仍可以為其手動設置一個名稱,比如:
#include <iostream>
using namespace std;
int main()
{
// display 即為 lambda 匿名函數的函數名
auto display = [](int a,int b) -> void{cout << a << " " << b;};
// 調用 lambda 函數
display(10,20); // 輸出:10 20
return 0;
}
可以看到,程序中使用 auto 關鍵字為 lambda 匿名函數設定了一個函數名,由此我們即可在作用域內調用該函數。
2.2 值傳遞和引用傳遞的區別
#include <iostream>
using namespace std;
// 全局變量
int all_num = 0;
int main()
{
// 局部變量
int num_1 = 1;
int num_2 = 2;
int num_3 = 3;
cout << "lambda1:\n";
auto lambda1 = [=] {
// 全局變量可以訪問甚至修改
all_num = 10;
// 函數體內只能使用外部變量,而無法對它們進行修改
cout << num_1 << " "
<< num_2 << " "
<< num_3 << endl;
};
lambda1();
cout << all_num << endl;
cout << "lambda2:\n";
auto lambda2 = [&] {
all_num = 100;
num_1 = 10;
num_2 = 20;
num_3 = 30;
cout << num_1 << " "
<< num_2 << " "
<< num_3 << endl;
};
lambda2();
cout << all_num << endl;
return 0;
}
程序執行結果為:
lambda1:
1 2 3
10
lambda2:
10 20 30
100
可以看到,在創建 lambda1 和 lambda2 匿名函數的作用域中,有 num_1、num_2 和 num_3 這 3 個局部變量,另外還有 all_num 全局變量。其中,lambda1 匿名函數是以 [=] 值傳遞的方式導入的局部變量,這意味著默認情況下,此函數內部無法修改這 3 個局部變量的值,但全局變量 all_num 除外。相對地,lambda2 匿名函數以 [&] 引用傳遞的方式導入這 3 個局部變量,因此在該函數的內部就可以訪問這 3 個局部變量,還可以任意修改它們。同樣,也可以訪問甚至修改全局變量。當然,如果我們想在 lambda1 匿名函數的基礎上修改外部變量的值,可以借助 mutable 關鍵字,例如:
auto lambda1 = [=]() mutable{
num_1 = 10;
num_2 = 20;
num_3 = 30;
// 函數體內只能使用外部變量,而無法對它們進行修改
cout << num_1 << " "
<< num_2 << " "
<< num_3 << endl;
};
由此,就可以在 lambda1 匿名函數中修改外部變量的值。但需要注意的是,這里修改的僅是 num_1、num_2、num_3 拷貝的那一份的值,真正外部變量的值并不會發生改變。
2.3 執行拋出異常類型
#include <iostream>
using namespace std;
int main()
{
auto except = []()throw(int) {
throw 10;
};
try {
except();
}
catch (int) {
cout << "捕獲到了整形異常"; // 輸出:捕獲到了整形異常
}
return 0;
}
可以看到,except 匿名數組中指定函數體中可以拋出整形異常,因此當函數體中真正發生整形異常時,可以借助 try-catch 塊成功捕獲并處理。
在此基礎上,再看一下反例:
#include <iostream>
using namespace std;
int main()
{
auto except1 = []()noexcept {
throw 100;
};
auto except2 = []()throw(char) {
throw 10;
};
try {
except1();
except2();
}
catch (int) {
cout << "捕獲到了整形異常" << endl;
}
return 0;
}
此程序運行會直接崩潰,原因很簡單,except1 匿名函數指定了函數體中不發生任何異常,但函數體中卻發生了整形異常;except2 匿名函數指定函數體可能會發生字符異常,但函數體中卻發生了整形異常。由于指定異常類型和真正發生的異常類型不匹配,導致 try-catch 無法捕獲,最終程序運行崩潰。
如果不使用 noexcept 或者 throw(),則 lambda 匿名函數的函數體中允許發生任何類型的異常。
原文鏈接:https://www.cnblogs.com/crossoverpptx/p/16932650.html
相關推薦
- 2022-07-21 A component required a bean of type ‘com.ruoyi.sea
- 2022-11-24 使用sqlserver官方驅動包調用存儲過程遇到的坑及解決方法_MsSql
- 2022-03-19 .NET6使WebApi獲取訪問者IP地址_基礎應用
- 2022-07-09 Android實現app開機自啟動功能_Android
- 2022-12-09 C++實現雙向起泡排序算法_C 語言
- 2022-03-18 c語言的指針數組詳解(c語言指針與數組)
- 2022-06-28 ES6基礎語法之對象介紹_基礎知識
- 2022-07-01 Go?函數中獲取調用者的函數名和文件名及行號_Golang
- 最近更新
-
- window11 系統安裝 yarn
- 超詳細win安裝深度學習環境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優雅實現加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發現-Nac
- Spring Security之基于HttpR
- Redis 底層數據結構-簡單動態字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支