当前位置: 首页 > news >正文

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

前言

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

基本语法

Syntaxindex
[ 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 <vector>
#include <iostream>
#include <algorithm>
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 <vector>
#include <iostream>
#include <algorithm>
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表达式,并且程序写起来更加方便,并不需要定义单独的打印函数。

相关文章:

  • 可能错误使用了‘offsetof’宏
  • 警告:对 NULL 对象非静态数据成员‘XXX::xxx’的访问无效
  • git tag常用操作
  • error: SEH exception with code 0xc0000005 thrown in the test
  • new对象数组是否会调用对象的构造函数
  • const究竟限制了谁的改变
  • CSDN 论坛板块升级规则
  • poj解题报告——序
  • UE4编辑器修改界面显示语言
  • poj解题报告——poj 2028 When Can We Meet?
  • poj解题报告——poj 1493 Machined Surfaces
  • poj解题报告——poj 2365 Rope
  • poj解题报告——poj 2575 Jolly Jumpers
  • poj解题报告——poj 3536 Beer Refrigerator
  • poj解题报告——poj 1528 Perfection
  • 《Javascript数据结构和算法》笔记-「字典和散列表」
  • 【mysql】环境安装、服务启动、密码设置
  • Android组件 - 收藏集 - 掘金
  • ES6, React, Redux, Webpack写的一个爬 GitHub 的网页
  • HomeBrew常规使用教程
  • Java多态
  • leetcode-27. Remove Element
  • Linux快速配置 VIM 实现语法高亮 补全 缩进等功能
  • Linux学习笔记6-使用fdisk进行磁盘管理
  • Promise面试题2实现异步串行执行
  • spark本地环境的搭建到运行第一个spark程序
  • spring boot下thymeleaf全局静态变量配置
  • spring security oauth2 password授权模式
  • vue-loader 源码解析系列之 selector
  • vue和cordova项目整合打包,并实现vue调用android的相机的demo
  • 动态规划入门(以爬楼梯为例)
  • 解析 Webpack中import、require、按需加载的执行过程
  •  一套莫尔斯电报听写、翻译系统
  • 你学不懂C语言,是因为不懂编写C程序的7个步骤 ...
  • #{}和${}的区别是什么 -- java面试
  • #NOIP 2014# day.1 T2 联合权值
  • #QT(TCP网络编程-服务端)
  • (12)Hive调优——count distinct去重优化
  • (1综述)从零开始的嵌入式图像图像处理(PI+QT+OpenCV)实战演练
  • (附源码)ssm教材管理系统 毕业设计 011229
  • (个人笔记质量不佳)SQL 左连接、右连接、内连接的区别
  • (汇总)os模块以及shutil模块对文件的操作
  • (简单) HDU 2612 Find a way,BFS。
  • (十)【Jmeter】线程(Threads(Users))之jp@gc - Stepping Thread Group (deprecated)
  • (一)Spring Cloud 直击微服务作用、架构应用、hystrix降级
  • (转载)hibernate缓存
  • (转载)虚幻引擎3--【UnrealScript教程】章节一:20.location和rotation
  • .NET Core 项目指定SDK版本
  • .Net Memory Profiler的使用举例
  • .NET Remoting Basic(10)-创建不同宿主的客户端与服务器端
  • .net分布式压力测试工具(Beetle.DT)
  • .NET面试题(二)
  • /var/log/cvslog 太大
  • ?
  • [【JSON2WEB】 13 基于REST2SQL 和 Amis 的 SQL 查询分析器