1, 定義
lambda 表達式就是一個函數(shù)(匿名函數(shù)),也就是一個沒有函數(shù)名的函數(shù)。為什么不需要函數(shù)名呢,因為我們直接(一次性的)用它,嵌入式用的它,不需要其他地方調(diào)用它。
lambda 表達式也叫閉包。閉就是封閉的意思(封閉就是其他地方都不調(diào)用它),包就是函數(shù)。
lambda 表達式其實就是一個函數(shù)對象,他內(nèi)部創(chuàng)建了一個重載()操作符的類。
lambda 表達式的簡單語法如下:[capture] (parameters) -> return value { body }, 只有 [capture] 捕獲列表和 { body } 函數(shù)體是必選的,其他可選。
2, 最簡單的一個 lambda 表達式(調(diào)用)
int main()
{
[] {}();//三部分,[] : 代表lambda表達式的開始;{} : 代表函數(shù)體,函數(shù)體里面什么都沒有;() : 代表函數(shù)調(diào)用.
}
等價于:
void f()
{
}
int main()
{
f();
}
像其他函數(shù)一樣,我們需要一個參數(shù)列表:()。所以:
[] {}(); 也可以寫為:
[](){}();
第二個位置加了一個()代表函數(shù)參數(shù)。如果什么參數(shù)都沒有,就可以省略 ()。
3, 再加點打印的東西:
#include <iostream>
using namespace std;
int main()
{
[] { cout << 'Hello, World!'<<endl; }();
}
也可以這樣:
#include <iostream>
using namespace std;
int main()
{
auto lam = [] { cout << 'Hello, World!'<<endl; };
lam();
}
4, 返回值
-> int :代表此匿名函數(shù)返回 int。大多數(shù)情況下lambda表達式的返回值可由編譯器猜測得出,因此不需要我們指定返回值類型。
int main()
{
auto lam =[]() -> int { cout << 'Hello, World!'; return 88; };
//auto lam =[]() { cout << 'Hello, World!'; return 88; };//自動推導返回值
auto ret = lam();
cout<<ret<<endl;//輸出88
auto lam2 =[]() -> string { cout << 'Hello, World!'; return 'test'; };
auto ret1 = lam2();
cout<<ret1<<endl;//輸出test
}
5, 捕捉變量
變量捕獲才是成就 lambda 卓越的秘方。
[] 不捕獲任何變量, 這種情況下lambda表達式內(nèi)部不能訪問外部的變量。
[&] 以引用方式捕獲所有變量
[=] 用值的方式捕獲所有變量(可能被編譯器優(yōu)化為const &)
[=, &foo] 以引用捕獲變量foo, 但其余變量都靠值捕獲
[&, foo] 以值捕獲foo, 但其余變量都靠引用捕獲
[bar] 以值方式捕獲bar; 不捕獲其它變量
[this] 捕獲所在類的this指針 (Qt中使用很多,如此lambda可以通過this訪問界面控件的數(shù)據(jù))
int a=1,b=2,c=3;
auto lam2 = [&,a](){//b,c以引用捕獲,a以值捕獲。
b=5;c=6;//a =1; a不能賦值
cout << a<<b<<c<<endl;//輸出 1 5 6
};
lam2();
6,
毋庸質(zhì)疑,lambda最大的一個優(yōu)勢是在使用STL中的算法 (algorithms) 庫時:
vector<string> address{'111','222',',333','.org','wwwtest.org'};
for_each(address.begin(),address.end(),[](const string& str){cout<<str<<endl;});
如此一行代碼就可以循環(huán)打印容器的數(shù)據(jù)。
再看一個例子,以前數(shù)組排序代碼(第二代排序,第一代是自己寫)是這樣的:
int arr[] = {6,4,3,2,1,5};
bool compare(int& a,int& b)//謂詞函數(shù)
{
return a>b;
}
std::sort(arr, arr+6, compare);
現(xiàn)在:
std::sort(arr, arr+6, [](const int& a,const int& b){return a>b;});//降序排序
//std::sort(arr, arr+6, [](const auto& a,const auto& b){return a>b;}); //C++14支持基于類型推斷的泛型lambda表達式。
std::for_each(begin(arr),end(arr),[](const int& e){cout<<'After:'<<e<<endl;});//6,5,4,3,2,1
lambda表達式取代了謂詞函數(shù),一行代碼搞定排序,再一行代碼搞定打印數(shù)組所有數(shù)據(jù)。
7, 打印類型
auto lam = [](){cout<<'hi'<<endl;};
cout<<typeid(lam).name()<<endl;//輸出 class <lambda_6c3f764f39072358acd689d114b4c204>
int i1;
cout<<'typeid:'<<typeid(i1).name()<<endl;//輸出 i
cout<<'typeid:'<<typeid(float).name()<<endl;//輸出 f
8,
創(chuàng)建lambda函數(shù)的一個原因是有些人創(chuàng)建了一個希望接受(lambda函數(shù))參數(shù)的函數(shù)。
lambda 的引入給我們帶來了一種全新的編程體驗,它可以讓我們把 “function” 當做是 “data” 一樣傳遞,并且使我們從繁瑣的語法中解放出來,更加關注于 “算法” 本身。
新的 std::function 是傳遞lambda函數(shù)的最好的方式,不管是傳遞參數(shù)還是返回值。
以下代碼將lambda表達式作為函數(shù)參數(shù)傳遞。程序的作用很簡單,是從一個地址簿中查找滿足條件的地址(匹配字符串或長度等規(guī)則)。( 參考: 「C++11」Lambda 表達式)
#include <iostream>
#include <string>
#include <vector>
#include <functional>
#include <algorithm>
using namespace std;
class AddressBook
{
public:
//提供一個通用的查找方法,以供查詢(匹配的地址),這個方法接受一個查找規(guī)則的函數(shù)作為參數(shù)
std::vector<string> findMatchingAddresses (std::function<bool (const string&)> func)
{
std::vector<string> results;
for ( auto it = _addresses.begin(), end = _addresses.end(); it != end; ++it ){
// 調(diào)用傳遞到findMatchingAddresses的函數(shù)并檢測是否匹配規(guī)則
if ( func( *it ) ){
results.push_back( *it );
}
}
return results;
}
void SetAddress(const std::vector<std::string> &address)
{
_addresses = address;
}
private:
std::vector<string> _addresses;
};
//聲明一個全局的AddressBook對象
AddressBook global_address_book;
//查找匹配名字的地址
vector<string> findAddressesFromName (const string &name)
{
return global_address_book.findMatchingAddresses(
[&] (const string& addr) { return addr.find(name) != string::npos; }
);
}
//查找匹配長度的地址(>min_len)
vector<string> findAddressesLen (const size_t &min_len)
{
return global_address_book.findMatchingAddresses( [&] (const string& addr) { return addr.length() >= min_len; } );
}
int main()
{
//初始化AddressBook對象的成員 _addresses
vector<string> address{'China chengdu','China hunan','TaiWan taibei','American alasijia','Japan dongjing'};
global_address_book.SetAddress(address);
//查找包含 China 的地址
auto ret = findAddressesFromName('China');
for_each(ret.begin(),ret.end(),[](string &i){cout<<i<<' ';});cout<<endl;//輸出: China chengdu 和 China hunan
//查找長度大于 15 的地址
auto ret2 = findAddressesLen(15);
for_each(ret2.begin(),ret2.end(),[](string &i){cout<<i<<' ';});cout<<endl;//輸出: American alasijia
return 0;
}
完。
版權聲明:本文為CSDN博主「宇宙379」的原創(chuàng)文章,遵循CC 4.0 BY-SA版權協(xié)議,轉載請附上原文出處鏈接及本聲明。