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

本文介绍了C++11中的Lambda表达式,一种新的匿名函数语法,可用于简化代码中仅需使用一次的小型函数。文章通过VS2008与VS2013的代码对比展示了Lambda表达式的便利性和简洁性。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

前言

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表达式,并且程序写起来更加方便,并不需要定义单独的打印函数。
在 Qt 中使用 `QTimer::singleShot` 时,可以非常方便地将 lambda 表达式作为参数传入,以实现一次性定时任务。该方法的语法形式如下: ```cpp QTimer::singleShot(int msec, const QObject *parent, Functor function); ``` 其中,`msec` 表示延迟执行的时间(单位为毫秒),`parent` 可用于指定父对象以管理生命周期,`function` 是要执行的函数或 lambda 表达式。 当使用 lambda 表达式时,需要注意捕获列表的设置,以及是否需要保持变量的有效性。例如,在以下示例中,使用了值捕获(`[=]`)方式来传递局部变量: ```cpp QString tmpImg = "~/Picture/xx.png"; QTimer::singleShot(1000, [=]() { QFile(tmpImg).remove(); }); ``` 上述代码中,`tmpImg` 被复制进 lambda 表达式的闭包中,确保其在 lambda 执行时仍然有效[^1]。此外,如果希望避免潜在的内存泄漏问题,也可以使用 `QPointer` 或者其他智能指针进行封装。 ### 使用带参数的 lambda 表达式 虽然 `QTimer::singleShot` 不直接支持带参数的 lambda 表达式,但可以通过绑定参数或者在 lambda 内部调用其他函数来间接实现。例如: ```cpp QString path = "testPath"; QTimer::singleShot(1000, [path]() { qDebug() << "File path to remove:" << path; QFile(path).remove(); }); ``` ### 注意事项 - **C++11 支持**:使用 lambda 表达式需要启用 C++11 标准支持,在 `.pro` 文件中添加 `CONFIG += c++11`。 - **捕获方式选择**:根据实际需求选择合适的捕获方式,如 `[=]`(值捕获)或 `[&]`(引用捕获)。注意引用捕获可能导致悬空引用问题。 - **线程安全**:若涉及跨线程操作,需确保访问的对象是线程安全的,或者通过 `Qt::QueuedConnection` 等机制调度到主线程执行。 ### 示例代码 下面是一个完整的示例,展示如何使用 `QTimer::singleShot` 和 lambda 表达式删除文件: ```cpp #include <QApplication> #include <QTimer> #include <QFile> #include <QDebug> int main(int argc, char *argv[]) { QApplication app(argc, argv); QString tmpImg = "/tmp/tempfile.txt"; // 创建一个临时文件用于演示 QFile(tmpImg).open(QIODevice::WriteOnly).close(); qDebug() << "File will be deleted after 3 seconds."; QTimer::singleShot(3000, [tmpImg]() { if (QFile::exists(tmpImg)) { QFile(tmpImg).remove(); qDebug() << "File removed."; } }); return app.exec(); } ``` 此代码将在 3 秒后删除指定路径下的文件,并输出相关信息[^2]。 ---
评论 34
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

AlbertS

常来“玩”啊~

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值