C++实用指南:Lambda 表达式的妙用
Lambda 表达式的灵活性和强大功能确实为编程提供了许多便利。但是我们发现许多开发者仍然无法灵活运用其便利,于是写了这篇文章。
Lambda 允许我们编写更简洁和灵活的代码。例如在处理网络请求时,我们经常需要确保响应与当前的状态或需求仍然相关。通过捕获上下文变量,Lambda 表达式可以帮助我们在处理异步回调时简化逻辑。
网络请求
案例1:通过捕获 i64RequestIndex 实现丢弃过期的请求回包
在第一个案例中,我们通过捕获请求索引 i64RequestIndex 来确保只处理最新的网络请求。如果请求索引与保存在对象中的最后一个请求索引相匹配,则我们可以确定这是最新的请求,并且可以安全地处理响应。如果索引不匹配,表明在此请求和响应之间,已经发起了新的请求,因此当前的响应已经过时,可以被丢弃。
// 成员变量:用于记录请求序号
int m_myLastRequestIndex = 0;// 发送网络请求并处理响应
uint64_t i64RequestIndex = ++m_myLastRequestIndex;
core::network::Send(request, [=](Rsp rsp) {if (this->m_myLastRequestIndex == i64RequestIndex) {// 只处理与最后一次请求索引匹配的响应}},nullptr);
案例2:合并请求以减少后台压力
第一个案例可能会发起很多无效请求。因此,在第二个案例中,我们不仅要检查请求是否为最新的,还需要考虑合并请求。当多个相似的请求在短时间内发起时,我们可以选择等待第一个请求的响应,然后根据需要决定是否发起新的请求。这种方式可以减轻服务器的压力,并提高应用程序的性能。
// 成员变量:用于跟踪是否有正在进行的请求
bool m_isRequestPending = false;
bool m_isNeedNewRequest = false;// 发起请求的包装函数
auto sendRequest = [&]() {m_isRequestPending = true;core::network::Send( request, [=](Rsp rsp) {m_isRequestPending = false;if (m_isNeedNewRequest ) {m_isNeedNewRequest = false;sendRequest();// 需要再次发起请求} else {// 处理最后一次请求的响应}},nullptr);
};// 逻辑判断是否需要发起请求
if (m_isRequestPending ) {// 如果当前已经有一个请求在进行,则等待这个请求的响应m_isNeedNewRequest = true;
} else {// 如果当前没有请求正在进行,则发起新的请求sendRequest();
}
在这个例子中,我们使用一个布尔变量 m_isRequestPending 来跟踪是否有请求正在进行。如果有请求正在进行,我们就等待该请求完成。在请求的回调中,我们将 isRequestPending 设置为 false 以表示请求已完成,并在必要时发起新的请求。
更多场景
1. 延迟执行(Lazy Evaluation)
Lambda 可以用来实现延迟计算,这允许代码仅在需要时才执行相关计算。这在优化性能和资源使用方面非常有用。
auto lazyValue = [expensiveComputation]() { return expensiveComputation(); };
// expensiveComputation 不会立即执行,直到调用 lazyValue()auto result = lazyValue(); // 在这里实际执行计算
2. 作为回调(Callbacks)
Lambda 经常用作回调函数,尤其是在 GUI 编程或事件驱动编程中。这允许开发者在同一处代码内部即定义事件行为,也提供了更好的上下文管理。
button->onClick([this](){ this->doSomething(); });
3. 作为函数对象(Functors)
Lambda 可以替代传统的函数对象(functors),简化语法并提高代码的可读性。
std::sort(vec.begin(), vec.end(), [](int a, int b) { return a < b; });
4. 作用域保护(Scope Guard)
Lambda 可以用来实现作用域保护模式,确保退出作用域时自动执行特定的清理代码。
auto guard = scope_guard([&] { cleanUpResources(); });
5. 实现装饰器模式(Decorator Pattern)
可以使用 Lambda 表达式来实现装饰器模式,动态地添加功能。
auto withLogging = [](auto func) {return [=](auto... args) {logBefore(args...);auto result = func(args...);logAfter(result);return result;};
};auto decoratedFunction = withLogging(someFunction);
6. 线程封闭(Thread Encapsulation)
在启动新线程时,Lambda 可以用来封装要在线程中运行的代码,从而使得创建线程的代码更加简洁。
std::thread t([=] { doWork(); });
t.join();
7. 实现状态机(State Machines)
Lambda 可以存储在容器中,使得状态转换和事件处理更加灵活。
std::map<State, std::function<void(Event)>> stateMachine;
stateMachine[State::INIT] = [](Event e) { /* 处理 INIT 状态的事件 */ };
8. 自定义迭代行为(Custom Iteration)
Lambda 可以与算法结合使用,以实现自定义的迭代行为。
std::for_each(collection.begin(), collection.end(), [](auto& item) { processItem(item); });
Lambda 表达式由于其匿名和内联特性,对于创建简洁、灵活的代码非常有用,它们已经成为现代C++编程中不可或缺的工具。
相关文章:
C++实用指南:Lambda 表达式的妙用
Lambda 表达式的灵活性和强大功能确实为编程提供了许多便利。但是我们发现许多开发者仍然无法灵活运用其便利,于是写了这篇文章。 Lambda 允许我们编写更简洁和灵活的代码。例如在处理网络请求时,我们经常需要确保响应与当前的状态或需求仍然相关。通过…...
FastAPI(七十八)实战开发《在线课程学习系统》接口开发-- 评论
源码见:"fastapi_study_road-learning_system_online_courses: fastapi框架实战之--在线课程学习系统" 梳理下思路 1.判断是否登录 2.课程是否存在 3.如果是回复,查看回复是否存在 4.是否有权限 5.发起评论 首先新增pydantic模型 class Cour…...
基于springboot+vue+uniapp的居民健康监测小程序
开发语言:Java框架:springbootuniappJDK版本:JDK1.8服务器:tomcat7数据库:mysql 5.7(一定要5.7版本)数据库工具:Navicat11开发软件:eclipse/myeclipse/ideaMaven包&#…...
TypeScript基础【学习笔记】
一、TypeScript 开发环境搭建 下载并安装 Node.js使用 npm 全局安装 typescript 进入命令行输入:npm i -g typescript 创建一个 ts 文件使用 tsc 对 ts 文件进行编译 进入命令行进入 ts 文件所在目录执行命令:tsc xxx.ts 二、基本类型 类型声明 通过类型…...
树莓派物联网服务器搭建流程:集成 Node.js、InfluxDB、Grafana 和 MQTT 协议
目录 一、搭建准备 1. 硬件要求 2. 软件要求 二、技术栈概述 三、搭建步骤 1. 安装操作系统 2. 启动树莓派 3. 更新系统 4. 安装必要的软件 4.1 安装 Python 和 Flask 4.2 安装 Node.js 4.3 安装 Mosquitto(MQTT Broker) 4.4 安装 InfluxDB…...
typescript 断言
/* 断言 确定后期一定是某种格式 可用于表单大对象初始化是空,赋值时有具体字段。前期断言是会是某种格式 */interface obj {name: stringlocation: stringage?: number }// 会报错 // let data: obj { // // } let data: obj {} as obj; // 断言data会有obj里…...
期刊评价指标及其查询方法
1、期刊评价体系一 科睿唯安《期刊引证报告》(Journal Citation Reports, JCR) 科睿唯安每年发布的《期刊引证报告》(Journal Citation Reports, JCR)是一个独特的多学科期刊评价工具。JCR数据库提供基于引文数据的统计信息的期…...
巴斯勒相机(Basler) ACE2 dart 系列说明和软件
巴斯勒相机(Basler) ACE2 dart 系列说明和软件...
【Pycharm中anaconda使用介绍】
在安装好anaconda之后,首先打开anaconda界面,执行以下操作 1.查看Anaconda中当前存在的环境 conda info -e 或者 conda-env list 查看–安装–更新–删除包 conda list: conda search package_name 查询包 conda install package_name conda …...
2024下半年,前端的技术风口来了
“ 你近期有体验过哪些大模型产品呢? 你有使用大模型API做过一些实际开发吗? 在你日常开发中,可以与大模型相关应用结合来完成工作吗? ” **最近,一直在和同事聊,关于前端可以用大模型干点啥ÿ…...
Spock Unit Test in Java
优质博文:IT-BLOG-CN 一、简介 Spock是一个基于Groovy语言的测试和规范框架,使得测试代码更简介,得益于JUnit Runner,Spock兼容大部分IDE和测试框架JUnit/JMock/Powermock等。基于BDD行为驱动开发,功能非常强大。提…...
:= 符号python
在 Python 3.8 及更高版本中,引入了一种新的语法特性,称为"海象运算符"(Walrus Operator),它使用 : 符号。这个运算符的主要目的是在表达式中同时进行赋值和返回赋值的值。 使用海象运算符可以在一些情况下…...
UPLOAD-LABS靶场[超详细通关教程,通关攻略]
---------------------------------------- 靶场环境: 下载链接: https://codeload.github.com/c0ny1/upload-labs/zip/refs/heads/master 使用小皮集成环境来完成这个靶场 将文件放到WWW目录下就可以进行访问 ------------------------------------…...
测试面试宝典(三十七)—— 接口测试中的加密参数如何处理?
1)先了解接口使用的加密方式(md5、rsa...) 2)检查接口测试工具是否支持这种加密方式,如果支持的话,直接使用对应功能就行了(比如Jmeter支持md5);如果加密方式是公司内部特有的算法,可以在接口测试工具中调…...
秋招突击——7/23——百度提前批面试准备和正式面试
文章目录 引言一面准备面试预演一1、讲一下hashcode()和equals()关系2、equals()和有什么区别3、讲一下重载和重写的区别4、讲一下深拷贝、浅拷贝的区别5、讲一下Java异常的基类,运行时异常举几个例子,什么情况下会出现?6、讲一下Java中线程的…...
学习日记:数据类型2
目录 1.转义字符 2.隐式类型转换 2.1 强制类型转换 2.2 不同类型间赋值 3.运算符 表达式 3.1 算术运算符 3.2 算术运算优先级 3.3 赋值运算 3.3.1 不同类型间混合赋值 3.4 逗号运算 4.生成随机数 5. 每日一练 1.转义字符 \n 表示换行 \t …...
Django Web框架——05
文章目录 admin 后台数据库管理注册自定义模型类修改自定义模型类的展现样式模型管理器类再谈Meta类 数据表关联关系映射一对一映射一对多映射多对多映射 cookies 和 sessioncookiessessionCookies vs session admin 后台数据库管理 django 提供了比较完善的后台管理数据库的接…...
【React】项目的目录结构全面指南
文章目录 一、React 项目的基本目录结构1. node_modules2. public3. src4. App.js5. index.js6. .gitignore7. package.json8. README.md 二、React 项目的高级目录结构1. api2. hooks3. pages4. redux5. utils 三、最佳实践 在开发一个 React 项目时,良好的目录结构…...
Django学习(二)
get请求 练习: views.py def test_method(request):if request.method GET:print(request.GET)# 如果链接中没有参数a会报错print(request.GET[a])# 使用这个方法,当查询不到参数时,不会报错而是返回你设置的值print(request.GET.get(c,n…...
Java引用类型
强软弱虚 以 ZGC 为例,谈一谈 JVM 是如何实现 Reference 语义的 SoftReference 到底在什么时候被回收 ? 如何量化内存不足 ? PhantomReference 和 WeakReference 究竟有何不同 ThreadLocal 和 Netty ByteBuf中使用到的引用类型 https://w…...
利用ngx_stream_return_module构建简易 TCP/UDP 响应网关
一、模块概述 ngx_stream_return_module 提供了一个极简的指令: return <value>;在收到客户端连接后,立即将 <value> 写回并关闭连接。<value> 支持内嵌文本和内置变量(如 $time_iso8601、$remote_addr 等)&a…...
让AI看见世界:MCP协议与服务器的工作原理
让AI看见世界:MCP协议与服务器的工作原理 MCP(Model Context Protocol)是一种创新的通信协议,旨在让大型语言模型能够安全、高效地与外部资源进行交互。在AI技术快速发展的今天,MCP正成为连接AI与现实世界的重要桥梁。…...
Android15默认授权浮窗权限
我们经常有那种需求,客户需要定制的apk集成在ROM中,并且默认授予其【显示在其他应用的上层】权限,也就是我们常说的浮窗权限,那么我们就可以通过以下方法在wms、ams等系统服务的systemReady()方法中调用即可实现预置应用默认授权浮…...
UR 协作机器人「三剑客」:精密轻量担当(UR7e)、全能协作主力(UR12e)、重型任务专家(UR15)
UR协作机器人正以其卓越性能在现代制造业自动化中扮演重要角色。UR7e、UR12e和UR15通过创新技术和精准设计满足了不同行业的多样化需求。其中,UR15以其速度、精度及人工智能准备能力成为自动化领域的重要突破。UR7e和UR12e则在负载规格和市场定位上不断优化…...
OpenLayers 分屏对比(地图联动)
注:当前使用的是 ol 5.3.0 版本,天地图使用的key请到天地图官网申请,并替换为自己的key 地图分屏对比在WebGIS开发中是很常见的功能,和卷帘图层不一样的是,分屏对比是在各个地图中添加相同或者不同的图层进行对比查看。…...
是否存在路径(FIFOBB算法)
题目描述 一个具有 n 个顶点e条边的无向图,该图顶点的编号依次为0到n-1且不存在顶点与自身相连的边。请使用FIFOBB算法编写程序,确定是否存在从顶点 source到顶点 destination的路径。 输入 第一行两个整数,分别表示n 和 e 的值(1…...
项目部署到Linux上时遇到的错误(Redis,MySQL,无法正确连接,地址占用问题)
Redis无法正确连接 在运行jar包时出现了这样的错误 查询得知问题核心在于Redis连接失败,具体原因是客户端发送了密码认证请求,但Redis服务器未设置密码 1.为Redis设置密码(匹配客户端配置) 步骤: 1).修…...
分布式增量爬虫实现方案
之前我们在讨论的是分布式爬虫如何实现增量爬取。增量爬虫的目标是只爬取新产生或发生变化的页面,避免重复抓取,以节省资源和时间。 在分布式环境下,增量爬虫的实现需要考虑多个爬虫节点之间的协调和去重。 另一种思路:将增量判…...
rnn判断string中第一次出现a的下标
# coding:utf8 import torch import torch.nn as nn import numpy as np import random import json""" 基于pytorch的网络编写 实现一个RNN网络完成多分类任务 判断字符 a 第一次出现在字符串中的位置 """class TorchModel(nn.Module):def __in…...
MySQL 8.0 事务全面讲解
以下是一个结合两次回答的 MySQL 8.0 事务全面讲解,涵盖了事务的核心概念、操作示例、失败回滚、隔离级别、事务性 DDL 和 XA 事务等内容,并修正了查看隔离级别的命令。 MySQL 8.0 事务全面讲解 一、事务的核心概念(ACID) 事务是…...
