C/C++开发基础——lambda表达式与std::bind闭包

本章主要内容:

一,lambda表达式

        1.基本概念

        2.关于捕获子句

        3.常见的捕获方式

二,闭包与std::bind模板

        1.什么是闭包

        2.std::bind的简介

        3.std::bind的用法

三,参考阅读



一,lambda表达式

C/C++开发基础——lambda表达式与std::bind闭包

1.基本概念

lambda表达式是从C++11开始引入的,主要用来定义匿名函数和闭包。lambda表达式可以被当作一个值赋给另一个变量,也可以作为实参传递给其他函数,或者作为其他函数的返回结果,用法类似于前面提到的函数对象和函数指针。如果只是把单个函数拿来传参,lambda表达式的使用方式比函数指针和函数对象更简洁。

lambda表达式可以不指定函数的返回类型,编译器将自动推导该类型。


lambda表达式的完整公式:

[capture_list](parameter_list) mutable -> return_type{ process code };
具体含义:
[]: lambda表达式的引出符,编译器根据该符号判断接下来的代码是否为lambda匿名函数。
parameter_list: 参数列表,与普通函数的参数列表一致。如果不需要传递参数,则可以省略该部分以及小括号()。
mutable: 使用了mutable修饰符的lambda表达式,不可以省略参数列表。
return_type: 函数返回值类型。该部分可以连同”->”一起省略。
process code: 函数体,它除了可以使用参数之外,还可以使用捕获到的变量。

lambda表达式样例: 
[](int x, int y){return x<y;}        //[]用来标记lambda表达式的开始
[](int x=0int y=0){return x<y;}    //传默认实参x=0,C++14标准开始支持
[]{return true;}                     //没有参数时,可以省略圆括号()
[](int x, int y)->bool{return x<y;}  //显式指定返回值类型,让代码更清晰

注意,lambda表达式中的”[ ]”不一定是空的,里面可以包含捕获子句,捕获子句用来捕获上下文中的变量来提供给lambda表达式使用。


C++代码样例: 

Demo1:

#include<iostream>
using namespace std;
int main() {
    auto operation = [](int a, int b, string op) -> double {
        if (op == "sum") {
            return a + b;
        }
        else {
            return (a + b) / 2.0;
        }
    };
    int num1 = 1;
    int num2 = 2;
    auto sum = operation(num1, num2, "sum");
    cout << "Sum = " << sum << endl;
    auto avg = operation(num1, num2, "avg");
    cout << "Average = " << avg;
    return 0;
}

运行结果:

Sum = 3
Average = 1.5

Demo2: lambda与std::for_each结合使用

#include <bits/stdc++.h>
#include <iostream>
using namespace std;
int main()
{
       vector<int> vec{ 12345 };
       for_each(vec.begin(),
                   vec.end(),
                   [](int& a) { a *= 2; }
               );
       for_each(vec.begin(),
                   vec.end(),
                   [](int a) { cout << a << " " ; }
                );
       cout << endl;
       return 0;
}

运行结果:

2 4 6 8 10

2.关于捕获子句

捕获子句定义了lambda表达式访问(捕获)表达式以外的参数和变量的方式。

默认的捕获子句有两种即”=”(按值捕获)和”&”(按引用捕获)。


为什么要有捕获子句:

当[ ]中为空时,lambda表达式只能访问lambda表达式中定义的局部实参和局部变量。当[ ]中不为空时,lambda表达式可以访问代码指定作用域中的所有参数和变量。因此,捕获子句的使用扩大了lambda表达式捕获变量的范围。


3.常见的捕获方式

方式一,按值捕获

方括号中包含”=”,指定作用域中变量的值可以传递到lambda表达式,lambda表达式可以使用变量的值,但是不能修改变量的值。


方式二,按引用捕获

方括号中包含”&”,指定作用域中变量的引用可以传递到lambda表达式,lambda表达式既可以使用变量的值,也可以修改变量的值。


方式三,捕获指定的变量

捕获变量和默认捕获子句的操作有些区别:

按值捕获变量:[ ]中直接传变量名,不带”=”。

按引用捕获变量:[ ]中传的是 “&”后面加变量名。

捕获多个变量时可以用逗号分隔,例如:

[=, &counter]  //按引用捕获counter,按值捕获其他变量
[&, counter]    //按值捕获counter,按引用捕获其他变量
指定的默认子句(“=”或”&”)必须是捕获列表中的第一项。
如果捕获列表的前面已经加了”=”捕获子句,则后面不能再按值捕获特定变量。同理,如果捕获列表的前面已经加了”&”捕获子句,则后面不能再按引用捕获特定变量。
所以下面这两个捕获子句会产生编译错误:
[&, &counter]
[=, &counter, number]
方式四,捕获this指针
如果一个对象的成员函数中有lambda表达式,那么这个lambda表达式不能通过按值捕获或按引用捕获这个对象的成员变量。为了让lambda表达式能够访问当前对象的成员变量,应该在捕获子句中使用this关键字。
有了this指针,lambda表达式可以访问当前对象的所有成员函数和成员变量,无论它们的访问权限被声明为protected还是private。

总结下来,常见的捕获语法有:
[=]: 按值捕获所有变量。
[&]: 按引用捕获所有变量。
[=,&x,&y]: 按引用捕获变量x和y,按值捕获其他变量。
[&,x,y]: 按值捕获变量x和y,按引用捕获其他变量。
[this]: 捕获当前的对象。
[*this]: 捕获当前的对象的副本。

C++代码样例

Demo1:
#include<iostream>
using namespace std;
int main() {
    int initial_sum = 100;
    auto add_to_sum = [initial_sum](int num) {
        return initial_sum + num;
    };
    int final_sum = add_to_sum(78);
    cout << "100 + 78 = " << final_sum;
    return 0;
}
运行结果:
100 + 78 = 178
Demo2:
#include <functional>
#include <iostream>
using namespace std;
int main()
{
    int i = 3;
    int j = 5;
    function<int(void)> f1 = [i, j] { return i + j; };
    i = 22;
    j = 44;
    function<int(void)> f2 = [&i, &j] { return i + j; };
    cout << f1() << endl;
    cout << f2() << endl;
}
运行结果:
8
66
Demo3:
#include <algorithm>
#include <iostream>
#include <vector>
using namespace std;
class Scale
{

public:
    explicit Scale(int scale) : _scale(scale) {}
    void ApplyScale(const vector<int>& v) const
    
{
        for_each(v.begin(), v.end(), [this](int n) { cout << n * _scale <<" "; });
        cout << endl;
    }
private:
    int _scale;
};
int main()
{
    vector<int> values;
    values.push_back(1);
    values.push_back(2);
    values.push_back(3);
    values.push_back(4);
    Scale s(3);
    s.ApplyScale(values);
}
运行结果:
3 6 9 12

二,闭包与std::bind模板

C/C++开发基础——lambda表达式与std::bind闭包

1.什么是闭包

闭包( Closure)这个概念起源于函数式编程,是指外部变量与函数之间的绑定,可以这样理解,捕获了外部变量的lambda表达式是一种闭包。


2.std::bind的简介

std::bind是C++11标准引入的函数模板,用于取代bind1st和bind2nd等旧式语法。std::bind常用来实现闭包,
它用于包装和调用特征相同的函数指针、函数对象或lambda表达式。
std::bind可以充当函数适配器,即它接受一个原函数作为输入并返回一个新的函数对象作为输出,返回的函数对象包含一个或多个与原函数绑定的参数。std::bind可以预先指定函数的所有参数,也可以将函数的部分参数预先指定好,剩下的参数等真正调用的时候再指定。

3.std::bind的用法

假如有一个计算两个数字相加的函数。
int add(int first, int second)
{
    return first + second;
}

std::bind将函数名作为其第一个参数,后面的参数用”_1,_2″这样的占位符来预留,得到一个函数对象add_func。add_func可以像函数一样直接被调用。

auto add_func = std::bind(&add, _1, _2);
add_func(4,5); //4+5, 返回9
假设遇到了特殊场景,需要将函数的第一个参数传12,第二个参数作为预留,使用方式如下。
auto new_add_func = std::bind(&add, 12, _1);
new_add_func(5); //12+5, 返回17

完整C++代码实现:

#include <iostream>
#include <functional>
using namespace std;
int add(int first, int second)
{
    return first + second;
}
int main()
{
    auto new_add_func = std::bind(&add, 12, placeholders::_1);
    int x = new_add_func(5);
    cout << x << endl;
    return 0;
}

运行结果:

17


三,参考阅读

C/C++开发基础——lambda表达式与std::bind闭包

《Beginning C++17, 5th Edition》

《深入理解C++11:C++11新特性解析与应用》

《C++高级编程》

https://leimao.github.io/blog/CPP-Closure/

https://thispointer.com/stdbind-tutorial-and-usage-details/

https://www.programiz.com/cpp-programming/lambda-expression

https://learn.microsoft.com/en-us/cpp/cpp/examples-of-lambda-expressions?view=msvc-170

原文始发于微信公众号(程序员与背包客):C/C++开发基础——lambda表达式与std::bind闭包

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

文章由极客之音整理,本文链接:https://www.bmabk.com/index.php/post/128581.html

(1)
小半的头像小半

相关推荐

发表回复

登录后才能评论
极客之音——专业性很强的中文编程技术网站,欢迎收藏到浏览器,订阅我们!