C++11(二):lamda表达式

前言

lamda表达式是c++11规范引入的新语法,我最早接触lamda表达式是在C#语言中,后来学习python的过程中渐渐发现这种语法的好处,实际上它就是一个匿名函数,如果你的代码里有一些只用了一次的小函数,不妨试试用lamda表达式来写一下。

基本语法

Syntax index
[ capture-list ] ( params ) mutable(optional) constexpr(optional)(c++17) exception attribute -> ret { body } (1)
[ capture-list ] ( params ) -> ret { body } (2)
[ capture-list ] ( params ) { body } (3)
[ capture-list ] { body } (4)

表格中表达式对应的含义:

  • (1)是一个完整的lamda表达式
  • (2)是一个const类型lamda表达式的定义,表达式不能改被复制捕获的(“capture”)列表中的值
  • (3)省略了返回值类型的 lambda 表达式,但是该 lambda 表达式的返回类型可以按照下列规则推演出来:

    • 如果 lambda 代码块中包含了 return 语句,则该 lambda 表达式的返回类型由 return 语句的返回类型确定。(直到C++14版本)
    • 如果没有 return 语句,则返回值为void,类似 void f(…) 函数。(直到C++14版本)
    • 对于一个返回auto类型的函数,返回值的类型由其返回的语句推断出来。(自从C++14版本)
  • (4)省略了参数列表,类似于无参函数 f(),这种形式只能用于constexpr, mutable, exception specification, attributes不存在的情况,或者末尾的返回值类型被使用的情况(感觉翻译过来有点问题,参考lamda表达式)。

语法分析:

mutable : 允许函数体修改被捕获的参数副本,并且可以访问被捕获对象的 non-const 方法。
constexpr:我拒绝解释它了,因为我渐渐的看到了一些关于他讨论,貌似被很多编译器拒绝了。。。
attribute :用来指定函数属性。
capture-list : 一个逗号分隔的列表。可以定义0个或多个捕获对象,具体示例如下:

  • [a,&b] : a以值的方式被捕获,b以引用的方式被捕获。
  • [this] : 以值的方式捕获this指针。
  • [&] : 以引用的方式捕获所有的外部自动变量。
  • [=] : 以值的方式捕获所有的外部自动变量。
  • [] : 不捕获任何变量。

params : 与命名函数中的参数列表一致,但是不允许使用默认参数(直到C++14版本),如果使用auto作为参数的类型,则这个lamda表达式将被看作是泛化的lamda表达式(自从C++14版本)。
ret : 返回值类型,如果不提供将会根据返回语句来推测(或者当完全无返回时,返回值为void)
body : 包含执行语句的函数体。

具体示例

1.VS2008 (代表C++03)遍历vector并打印值的代码:

#include 
#include 
#include 
using namespace std;

// using c++03 standard
void print_value(int value)
{
    std::cout << value << std::endl;
}

int main()
{
    std::vector<int> v;
    v.push_back(1);
    v.push_back(2);
    v.push_back(3);

    std::for_each(v.begin(), v.end(), print_value);
    return 0;
}

2.VS2013 (代表C++11)遍历vector并打印值的代码:

#include 
#include 
#include 
using namespace std;

// using c++11 standard
int main()
{
    std::vector<int> v;
    v.push_back(1);
    v.push_back(2);
    v.push_back(3);

    std::for_each(std::begin(v), std::end(v), [](int n) {std::cout << n << std::endl;});
    return 0;
}

总结

  • lamda表达式起到函数的作用。
  • 代码中需要一些小函数,但是只需要一次的时候,就可以定义成lamda表达式,可以使程序更简洁。
  • 通过上面的对比可以发现,C++11支持lamda表达式,并且程序写起来更加方便,并不需要定义单独的打印函数。

你可能感兴趣的:(C++,C++11/17/20新特性,IDE,lamda,C++11,匿名函数,C++,capture)