1. 程式人生 > >C++ lambda表示式入門

C++ lambda表示式入門

1.lambda表示式 lambda表示式 是一個函式,一個匿名函式,也就是沒有函式名的函式,為什麼不需要函式名呢,因為我們直接(一次性的)用它,嵌入式用的它,不需要其他地方用它。 也叫閉包,閉就是封閉的意思,就是其他地方都不用他,包就是函式。 lambda表示式 其實就是一個函式物件,他內部建立了一個過載()操作符的類。

最簡單的一個lambda表示式。

int main()
{
[] {}();//[]代表lambda表示式的開始,{}代表函式體,什麼都沒有,()代表呼叫函式.
}

等價與:

void f()
{
	
}
int main()
{
	f();
}

像其他函式一樣,我們需要一個引數列表:()。所以: [] {}(); 也可以寫:

 []() {}(); 

加了一個()代表函式引數。什麼引數都沒有,就可以省略。 3. 再加點列印的東西:

int main()
{
	 [] { cout << "Hello, World!"; }();
}

也可以這樣:

int main()
{
	auto lam = [] { cout << "Hello, World!"; };
	lam();
}
  1. 返回值 -> int :代表返回int。
 int main()
{
	auto lam =[]() -> int { cout << "Hello, World!"; return 1; };
    auto ret = lam();
	auto lam2 =[]() -> string { cout << "Hello, World!"; return "test"; };
    auto ret1 = lam2();
}
  1. 捕捉變數 捕捉變數是變數捕獲才是成就lambda卓越的祕方。

[] 不捕獲任何變數 [&] 以引用方式捕獲所有變數 [=] 用值的方式捕獲所有變數(可能被編譯器優化為const &) [=, &foo] 以引用捕獲foo, 但其餘變數都靠值捕獲 [&, foo] 以值捕獲foo, 但其餘變數都靠引用捕獲 [bar] 以值方式捕獲bar; 不捕獲其它變數 [this] 捕獲所在類的this指標

int a=1,b=2,c=3;
auto lam2  =[&,a](){ cout << a<<b<<c<<endl;};//b,c以引用捕獲,a以值捕獲。
lam2();

毋庸質疑,lambda最大的一個優勢是在使用STL中的演算法(algorithms)庫時。

vector<string> address{"111","222",",333",".org","wwwtest.org"};
for_each(address.begin(),address.end(),[](string& str){cout<<str<<endl;});

如此一行程式碼就可以迴圈列印容器的資料。

建立lambda函式的一個原因是有些人建立了一個希望接受lambda函式的函式。 新的std::function是傳遞lambda函式的最好的方式,不管是傳遞引數還是返回值。 ( 參考: https://www.cnblogs.com/kedebug/p/3224561.html)

#include <iostream>
#include <string>
#include <vector>
#include <functional>
#include <algorithm>
using namespace std;

class AddressBook
{
public:
    std::vector<string> findMatchingAddresses (std::function<bool (const string&)> func)
    {
        std::vector<string> results;
        for ( auto itr = _addresses.begin(), end = _addresses.end(); itr != end; ++itr )
        {
            // 呼叫傳遞到findMatchingAddresses的函式並檢測是否匹配規則
            if ( func( *itr ) )
            {
                results.push_back( *itr );
            }
        }
        return results;
    }
    void SetAddress(const std::vector<std::string> &address)
    {
        _addresses = address;
    }
private:
    std::vector<string> _addresses;
};

AddressBook global_address_book;
//查詢匹配名字的地址
vector<string> findAddressesFromName (const string &name)
{
    return global_address_book.findMatchingAddresses(
                [&] (const string& addr) { return addr.find(name) != string::npos; }
    );
}
//查詢匹配長度的地址
vector<string> findAddressesLen (const int &min_len)
{
    return global_address_book.findMatchingAddresses( [&] (const string& addr) { return addr.length() >= min_len; } );
}

int main()
{    
    vector<string> address{"china chengdu","china hunan","taiwan taibei","american alaisjia","riben dongjing"};
    global_address_book.SetAddress(address);

    auto ret = findAddressesFromName("china");
    for_each(ret.begin(),ret.end(),[](string &i){cout<<i<<",";});cout<<endl;

    auto ret2 = findAddressesLen(15);
    for_each(ret2.begin(),ret2.end(),[](string &i){cout<<i<<",";});cout<<endl;

    return 0;
}