C++:函数对象:Lambda:Lambda详解(三)

news2025/1/17 21:42:45

1:定义

  • lambda表达式就是一个函数(匿名函数),也就是一个没有函数名的函数。为什么不需要函数名了? 因为我们直接(一次性的)用它,不需要其他地方调用它。
  • lambda表达式也叫闭包,闭就是封闭的意思(就是其他地方都不调用它),包就是函数
  • lambda表达式其实就是一个函数对象,它内部创建了一个重载()操作符的类
  • lambda表达式的简单语法如下:
  • [capture] (paramters) ->return value {body},只有[capture] 捕获列表和 {body} 函数体是必选的,其他可选。

2:最简单的一个 lambda表达式(调用) 

int main() 
{
    // [] 代表 lambda表达式的开始
       () 代表 函数调用
       {} 代表 函数体 

    // lambda表达式
    [] () {} 
}


// 上述表达式等价于

void f()
{

}

int main()
{
    f();
}

3: lambda表达式调用 

#include<iostream>
#include<string>

using namespace std;

int main()
{
    // 省略 表达式的 形参和返回值
    [] {cout << "hello world"<< endl;};

    // 或者这样写
    auto lam = [] {cout << "hello world"<< endl;};
    lam();
}

4: 返回值 

-> int : 代表此匿名函数返回 int  

 大多数情况下  lambda 表达式的返回值可由编译器猜测出阿里,因此我们不需要指定返回值类型。

int main()
{
    // 编译器可以自动推导出 函数体返回值,可以省略 返回int 返回类型
    auto lam = []() -> {cout << "hello,world"; return 88;};

    // 等价于
    auoto lam2 = [](){cout << "hello,world"; return 88;};

    auto ret = lam();
    cout << ret<< endls;  // 打印 88

    auto lam3 = [](){cout<< "hello world"; return "test str"};
    auto ret3 = lam3();
    cout << ret3 << endls;  // 打印 test str
}

5: 捕捉变量 

变量捕获才是  lambda最卓越的秘方。

  1. [ ]   ----- 不捕获任何变量,这种情况下 lambda 表达式内部不能访问外部变量
  2. [&]  -----以引用方式捕获所有变量
  3. [=] ------用值得方式捕获所有变量 (也有可能被编译器优化为 const &)

[=, &foo]  ----以引用的方式捕获 foo 变量,其他变量通过值方式捕获

[&, foo]  ---- 以值方式捕获 foo 变量,其他变量靠引用捕获

[bar] ------以值方式捕获bar变量,不捕获其他变量

[this] ------捕获所在类的this 指针 (在QT中使用的比较多,如此lambda可以通过this方位界面控件的数量。)

int main()
{
    int a = 1,b=2,c=3;
    auto lam2 = [&,a] ()
    {

        // a以值得方式捕获,b和c以引用方式捕获
        b = 4, c= 5;
        a = 6;  // error : a是不可修改的左值
        
        // 打印输出:1 4 5
        cout<< a << b << c<<enld;  
    };
    lam2();
}

6: lambda 在 STL中算法库中应用

毋庸置疑:  lambda表达式最大的一个优势是在 使用 STL中的算法(algorithms)库应用。

案例1:循环打印容器中的数据

vector<string> address = {"1111","222","333","org","www"};

int mian()
{
    for_each(address.begin(), address.end(),[](const string& str){

         cout<< str << endl;
    });
}

案例2: 给数组排序

// 数组代码排序(自己写一个降序接口)

int arr[] = {6,10,8,9,2,5};

bool compare(int& a, int& b) {
    // 降序排序
    return a>b;
}

std::sort(arr,arr+6 ,compare);



// 给数组排序(lambda表达式)
str::sort(arr,arr+6, [](const int& a, const int& b){
        return a>b; 
});

// 优化 C++14 支持基类类型推导的泛型
str::sort(arr,arr+6, [](const auto& a, const auto& b){
        return a>b; 
});

// 打印排序后的数据
std::for_each(begin(arr),end(arr), [](const int& a) {
    cout<< "After sort: " << a;
});

7:  创建lambda函数的一个原因是:有些人创建了一个希望接受(lambda函数)参数的函数

  1. lambda 的引入给我吗带来一种全新的编程体验,它可以让我们把  "function" 当做是“data”一样传递,并且使我们从繁琐的语法中解放出来,更加关注 “算法”本身。
  2. 新的 std::function 是传递lambda函数的最好方式,不管是传参数还是传返回值
  3. 以下代码:将lambda表达式作为函数参数传递,程序作用很简单,是从一个地址薄中查找满足条件的地址(匹配字符串或者长度规则)
#include<iostream>
#include<string>
#include<vector>
#include<functional>
#include<algorithm>

using namespace std;

class AddressBook
{
public:
	// 提供一个通用的查找方法,以供查询(匹配的地址),这个方法接受一个查找规则的函数作为参数。
	std::vector<string> findMatchingAddress(std::function<bool(const string&)> func)
	{
		std::vector<string> results;
		for (auto it = _addresses.begin(),end = _addresses.end();it != end; ++it)
		{
			// 调用传递到 findMatchingAddresses的函数并检测是否匹配规则
			if (func(*it))
			{
				results.push_back(*it);
			}
		}
		return results;
	}

	void SetAddress(const std::vector<std::string>& address) {
		_addresses = address;
	}

private:
	std::vector<string> _addresses;
};

// 声明一个全局的 AddressBook对象。
AddressBook global_address_book;

// 查找匹配名字的地址
vector<string> findAddressesFromName(const string& name)
{
	return global_address_book.findMatchingAddress(
		[&](const string& addr) { 
			return addr.find(name) != string::npos; 
		}
	);
}

// 查找匹配长度的地址(大于 min_len)
vector<string> findAddressLen(const size_t& min_len) {
	return global_address_book.findMatchingAddress(
		[&](const string& addr) {
			return addr.length()>= min_len;
		}
	);
}

int main() {
	// 初始化 AddressBook 对象的成员 _addresses
	vector<string> address{ "china chengdu","china wuhan",
		"china guangzhou","japan dongjing","Americal huashengdun"};
	global_address_book.SetAddress(address);

	// 查找包含china的地址
	auto ret = findAddressesFromName("china");
	for_each(ret.begin(), ret.end(), [](string& str) {
		cout << str << " ";
		});
	cout << endl;

	// 查找长度大于15的地址
	auto ret2 = findAddressLen(15);
	for_each(ret2.begin(), ret2.end(), [](string& len) {
		cout << len << " ";
		});

}

 请注意看下面这个问题:std::function<> func : 泛型返回值和入参均要和实参对应。

 

 

 

 

 

 

 

 

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/158671.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

K8s部署前后端分离项目(二)

K8s容器部署两个项目&#xff08;主节点操作&#xff09; 1、创建yaml文件 master节点上创建todo-list.yaml文件&#xff0c;两个jar包设置了两个副本&#xff0c;vue设置了一个副本&#xff0c;端口号固定配置。 当前文件目录为&#xff1a;/home/docker 创建yaml文件命令为&…

数学建模----图与网络模型

目录 一.图的基本概念与数据结构 1.基本概念 2.图与网络的数据结构 1.邻接矩阵表示法 2.关联矩阵 3.Matlab工具箱简介 1.图的生成 4.问题讨论 1.最短路问题 2.最小生成树问题 一.图的基本概念与数据结构 1.基本概念 点对应于研究对象&#xff0c;根据关系将一些点对应相…

第14章 多数据库支持之MySql

对于一个工程性程序而言&#xff0c;是否支持对多数据库的CURD操作&#xff0c;不是必须的&#xff0c;只是为了预防程序部署和运行过程中如果因为版权问题而产生的额外成本而预留的退路。对于移动设备的前后端分离程序而言&#xff0c;如果数据库需要部署到移动设备中&#xf…

【UE4 第一人称射击游戏】42-消耗和重新装载“M4A1”弹药

上一篇&#xff1a;【UE4 第一人称射击游戏】41-让“M4A1”拥有独立的弹药系统本篇效果&#xff1a;可以看到当玩家拾取了“M4A1”后&#xff0c;实现了重新装弹的功能&#xff0c;与“AK47”的弹药系统是相互独立的。步骤&#xff1a;打开“ThirdPersonCharacter”&#xff0c…

PHP session反序列化漏洞原理解析

什么是session 官方Session定义&#xff1a;在计算机中&#xff0c;尤其是在网络应用中&#xff0c;称为“会话控制”。Session对象存储特定用户会话所需的属性及配置信息。主要有以下特点&#xff1a; session保存的位置是在服务器端 session通常是要配合cookie使用 因为HTTP…

Docker Swarm NFS 数据持久化存储

目录一、Swarm 集群部署二、NFS 服务部署三、Swarm 使用 NFS3.1 通过 Volume3.1.1 创建 Volume3.1.2 使用 Volume3.1.3 验证数据共享特性3.2 通过 Docker Stack3.2.1 创建 YAML 文件3.2.2 使用 YAML 文件3.2.3 验证数据共享特性一、Swarm 集群部署 可参考我前面的博客《基于 L…

从数据到价值,DataOps精益数据运营概述

作者&#xff1a;陈荣耀 阿里云全球技术服务团队 一、背景&挑战 数字化时代&#xff0c;企业希望借助数字化的技术能力来提升企业的经营能力&#xff0c;从最终业务目标上来看&#xff0c;一般分三类&#xff1a; 1. 增加收入&#xff1a;基于经营数据的智能分析来提升产…

24、TORCH.UTILS.DATA

PyTorch 数据加载实用程序的核心是 torch.utils.data.DataLoader 类。它代表一个可在数据集上迭代的 Python&#xff0c;支持map-style and iterable-style datasets,customizing data loading order,automatic batching,single- and multi-process data loading,automatic mem…

CTK Plugin Framework插件框架学习--服务追踪

文章目录一、前言二、新建插件PluginA三、新建插件PluginB四、测试一、前言 服务追踪&#xff1a;如果想在B插件里使用A服务&#xff0c;可以专门写一个类继承ctkServiceTracker&#xff0c;在这个类里完成对A服务的底层操作&#xff0c;然后在B插件里通过这个类提供的接口来使…

重庆市市长胡衡华会见深兰科技董事长陈海波一行

1月9日&#xff0c;重庆市市长胡衡华会见了赴渝考察调研的深兰科技集团创始人、董事长陈海波一行&#xff0c;双方就开展互利合作进行了深入交流。胡衡华市长会见深兰科技考察团重庆市委常委、副市长陈鸣波&#xff0c;市政府秘书长、办公厅主任欧顺清&#xff0c;市政府副秘书…

ResT: An Efficient Transformer for Visual Recognition

文章地址: https://arxiv.org/pdf/2105.13677.pdf codeResT: An Efficient Transformer for Visual Recognition一、引言二、ResT一、Transformer模块的再思考二、Efficient Transformer Block三、Patch Embedding四、Positional Encoding五、整体架构三、实验一、分类二、目标…

go import package 入门lib1 is not in GOROOT

main.go:4:2: package lib1 is not in GOROOT (/usr/local/go/src/lib1)├── 5-init│ ├── lib1│ │ └── lib1.go│ └── lib2│ └── lib2.go├── const.go├── firstVar.go├── go.mod├── helloGolang.go├── main.go└── test3function.gogo env …

Java学习之单例设计模式

目录 设计模式 单例模式 一、饿汉式 二、懒汉式 三、饿汉式VS懒汉式 总结 设计模式 1.静态方法和属性的经典使用 2.设计模式是在大量的实践中总结和理论化之后优选的代码结构、编程风格以及解决问题的思考方式。就像是经典的棋谱&#xff0c;不同的棋局&#xff0c;我们用…

连接查询之外连接(左外链接和右外连接)

内连接&#xff1a; 假设A表和B表进行连接查询&#xff0c;使用内连接的话&#xff0c;凡是A表和B表能够匹配上的记录被查询出来&#xff0c;这就是内连接。A、B两张表没有主副之分&#xff0c;两张表是平等的。 外连接&#xff1a; 假设A表和B表进行连接查询&#xff0c;使用…

对于html中div标签height属性的个人理解

对于没有系统学习过css的程序员来说&#xff0c;在编写css样式的时候&#xff0c;div的height属性值确实是个玄学的东西&#xff0c;我也感觉css确实挺玄学的&#xff0c;本文将介绍我对div标签height属性的个人理解&#xff0c;如有问题请指正。 在html中&#xff0c;div标签属…

xilinx srio ip学习笔记之srio example

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 xilinx srio ip学习笔记之srio example前言IP的配置例程前言 前面对SRIO的理论有了初步的理解&#xff0c;现在急需要通过理解例程并且修改例程来建立自信心了。 学东西确实…

Java版本TransH代码的学习

主要讲和TransE代码的区别&#xff0c;TransE文章的链接 Java版本TransE代码的学习 关于范数的概念 什么是0范数、1范数、2范数&#xff1f;区别又是什么 初始化向量 初始化关系平面的向量Wr&#xff0c;初始化向量relation_vec,初始化节点向量entity_vec Wr_vec new doub…

富豪酒店集团全新体验「METAGREEN」上线!边玩边赚,了解可持续发展!

富豪酒店集团推出 MetaGreen 以提高大众对可持续发展的认识&#xff0c;并创造一个多元化的绿色生态系统。 体验将包涵盖数个独特的互动地标&#xff0c;包括环保富豪酒店、大华银行艺术空间、恒生银行元宇宙分行&#xff0c;以及 citysuper、LOG-ON 和 The Mills 等零售商。 …

擎创技术流 | ClickHouse实用工具—ckman教程(9)

哈喽~大家好&#xff0c;时间倏然&#xff0c;上一次ckman分享还是在2022&#xff0c;这一期分享就已经是2023了。由于前段时间小编“成功加入羊群”&#xff0c;导致拖更一周&#xff0c;实在抱歉。希望新的一年大家都可以身体健健康康&#xff0c;事业红红火火&#xff0c;生…

Clarifying Question领域最常见的三个数据集

文章目录Qulacqulac.json:qulac_hist012_dict.tar.gz:MIMICSClariQConvAI3 Data ChallengeStage1: initial datasetStage2: human-in-the-loopClariQ DatasetFile Formattrain.tsv and dev.tsvtest.tsvquestion_bank.tsvdev_synthetic.pkl.tar.gz & train_synthetic.pkl.ta…