c++11 lambda表达式

lambda表达式定义了一个匿名函数,并且可以捕获一定范围内的变量。

形如:[capture](params) opt->ret{body;}

capture是捕获列表,params是参数列表,opt是函数选项,如mutable,ret是函数返回值类型,body是函数体

[]不捕获任何变量

[&]按引用捕获作用域中的所有变量

[=]按值捕获作用域中的所有变量

[=,&f]按引用捕获f变量,其它变量按值捕获

[f]按值捕获f变量,其它变量不捕获

[this]捕获当前类中的this指针,如果已经使用了&或者=,就默认添加此选项。捕获this的目的是可以在lamda中使用当前类的成员函数和成员变量

class A
{
public:
    int i = 0;

    void func(int x, int y)
    {
        //auto x1 = [] { return i; };            //error,没有捕获外部变量
        auto x2 = [=] { return i + x + y; };
        auto x3 = [&] { return i + x + y; };
        auto x4 = [this] { return i; };
        //auto x5 = [this] { return i + x + y; };        //error,没有捕获x,y
        auto x6 = [this, x, y] { return i + x + y; };
        auto x7 = [this] { return i++; };
    }
};

需要注意的是,按值捕获的外部变量是无法修改的,如果要修改这些变量,需要按引用捕获。如果
仍要按值捕获外部变量并修改变量,这时,需要显式指明lambda表达式为mutable,而被mutable修饰的lambda表达式就算没有参数也要写明参数列表。

    int a = 0;
    //auto f1 = [=] { return a++; };                //error
    auto f2 = [=]() mutable {return ++a; };            //ok,f2()返回1,a = 0

lambda表达式的类型在c++11被称为“闭包类型”,我们可以使用std::function和std::bind来存储操作lambda表达式。对于没有捕获任何变量的lambda表达式,还可以被转换成一个普通的函数指针:

using func = int(*)(int);
    func f = [](int a){
        return a;
    };

### 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、付费专栏及课程。

余额充值