【C++11】lambda表达式

目的

  1. 匿名函数更加直观,想要了解一个函数的内部实现不用再从主函数中的函数调用再去观察函数体
  2. 简洁灵活,避免了代码膨胀和功能分散

表示方式

[capture](params) opt -> ret {body;};
//capture用于捕获外部变量,即不属于匿名函数中的变量,按值捕获和按引用捕获
//params表示匿名函数形参
//opt:一共两种,mutable可以修改按值传递进来的拷贝,exception抛异常
//ret:返回值类型
//body:函数体

细节

  1. 按值捕获的变量是只读的,加上mutable后,可改外可读可写的

为什么按值捕获的变量是只读的?

  1. lambda表达式的类型在C++11中会被看做是一个带operator()的类,即仿函数。
  2. 按照C++标准,lambda表达式的operator()默认是const的,一个const成员函数是无法修改成员变量值的。
  3. mutable 选项的作用就在于取消 operator () 的 const 属性。

作者: 苏丙榅
链接: https://subingwen.cn/cpp/lambda/#4-%E5%87%BD%E6%95%B0%E6%9C%AC%E8%B4%A8
来源: 爱编程的大丙
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

  1. 匿名函数大括号后加上小括号才表示调用匿名函数,小括号内部传入实参
  2. 不指定 lambda 表达式的返回值,编译器会根据 return 语句自动推导返回值的类型,但需要注意的是 labmda表达式不能通过列表初始化自动推导出返回值类型。
    因为初始化列表能够初始化的类型有很多,如{1,2,3}既可以是初始化数组,也可以是初始化结构体,因此此时是需要写明返回值的 -> array

建议

举例

//匿名函数,更直观灵活
//实质是一个仿函数
#include <bits/stdc++.h>
using namespace std;
void func(int x, int y){
    int a;
    int b;
    [=]()mutable{//按值捕获时外部变量是只读的,若加上mutable可选项,可变为可读可写的
        int c = a;
        x++;
        int d = x;
        cout << "内部:" << x << endl;
    }();//大括号后加上小括号表示匿名函数被调用,括号内是实参
    cout << "外部:" << x << endl;
}
int main(){

    func(2, 3);
    system("pause");
    return 0;
}

运行结果

在这里插入图片描述

参考学习

https://www.bilibili.com/video/BV1bX4y1G7ks

### C++11 Lambda表达式使用教程 #### 定义与基本语法 C++11引入了lambda表达式这一强大特性,使得开发者能够更加便捷地定义和创建匿名函数。这种新的功能不仅简化了代码编写过程,还提升了程序的功能性和执行效率[^1]。 Lambda表达式的通用形式如下: ```cpp [capture](parameters)->return_type { body } ``` 其中`capture`表示捕获列表,用于指定如何访问外部作用域中的变量;`parameters`代表参数列表;`->return_type`可选部分用来显式声明返回类型;最后是函数体`body`[^2]。 #### 捕获列表详解 捕获列表允许lambda表达式获取其所在环境内的局部变量或全局变量。常见的几种方式包括但不限于按传递(`[var]`)、按引用传递(` [&var] `),以及默认全部按/引用捕获(` [= ] / [&] `)[^3]。 ##### 示例:按捕获 当采用按的方式捕获时,意味着会复制一份原始数据供内部逻辑操作而不影响原。 ```cpp #include <iostream> using namespace std; int main(){ int value = 42; auto func_by_value = [value]() { cout << "Value captured by copy: " << value << endl; }; ++value; // 修改外界的value func_by_value(); // 输出的是未修改前的数 } ``` ##### 示例:按引用捕获 相反地,如果选择了按引用的形式,则任何对于被捕获变量的操作都会直接影响到实际存在的那个实例。 ```cpp #include <iostream> using namespace std; int main(){ int ref_val = 88; auto modify_ref = [&ref_val]() mutable{ ++ref_val; cout << "Modified reference-captured variable to : " << ref_val << endl; }; modify_ref(); } ``` #### 实际应用场景举例 考虑这样一个场景——我们需要交换两个整型数的位置而无需借助额外的空间开销。此时可以巧妙运用带有适当捕获机制的lambda表达式来完成任务[^5]。 ```cpp void swapTwoNumbers(int &num1, int &num2){ auto swapper = [&](int &a, int &b){ int temp=a;a=b;b=temp; }; swapper(num1,num2); } // 或者更为简洁的做法直接在lambda体内处理 auto direct_swap=[&](int &x,int &y){swap(x,y);} direct_swap(a,b); ``` #### 关于赋行为的重要注意事项 得注意的一点在于,尽管可以通过拷贝构造的方式来生成另一个具有相同行为的新对象,但是不同lambda表达式之间不允许互相赋,即便它们看起来拥有完全一致的结构和语义[^4]。 ```cpp auto hello_world_1=[](){std::cout<<"Hello World"<<std::endl;}; auto hello_world_2=[](){std::cout<<"Hello World"<<std::endl;}; // 下面这行会导致编译错误 //hello_world_1=hello_world_2; ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值