匿名函数lambda
[捕获列表](参数列表) mutable(可选) 异常属性 -> 返回类型 {
// 函数体
}
语法规则:lambda表达式可以看成是一般函数的函数名被略去,返回值使用了一个 -> 的形式表示。唯一与普通函数不同的是增加了“捕获列表”。
// lambda_test lambda_test.cc
#include <iostream>
using namespace std;
void test01()
{
cout << "test01" << endl;
auto Add = [](int a, int b) -> int {
return a + b;
};
cout << Add(1, 2) << endl;
}
int main(int argc,char **argv)
{
test01();
return 0;
}
编译(要指定-std=c++11):
g++ -o lambda_test lambda_test.cc -std=c++11
输出结果:
$ ./lambda_test
test01
3
一般情况下,编译器可以自动推断出lambda表达式的返回类型,所以我们可以不指定返回类型,即:
// lambda_test lambda_test.cc
#include <iostream>
using namespace std;
void test02()
{
cout << "test02" << endl;
auto Add = [](int a, int b){
return a + b;
};
cout << Add(1, 2) << endl;
}
int main(int argc,char **argv)
{
//test01();
test02();
return 0;
}
但是如果函数体内有多个return语句时,编译器无法自动推断出返回类型,此时必须指定返回类型。
有时候,需要在匿名函数内使用外部变量,所以用捕获列表来传递参数。根据传递参数的行为,捕获列表可分为以下几种:
与参数传值类似,值捕获的前提是变量可以拷贝,不同之处则在于,被捕获的变量在 lambda表达式被创建时拷贝,而不是在调用时才拷贝:
// lambda_test lambda_test.cc
#include <iostream>
using namespace std;
void test03()
{
cout << "test03" << endl;
int c = 20;
int d = 30;
auto Add = [c,d](int a, int b) {
cout << "d = "<< d << endl;
return c;
};
d = 10; // 在这里修改 d 的值,会改变 Add里的 d 值吗?
cout << Add(1, 2) << endl;
}
int main(int argc,char **argv)
{
//test01();
//test02();
test03();
return 0;
}
执行结果:
$ ./lambda_test
test03
d = 30
20
与引用传参类似,引用捕获保存的是引用,值会发生变化。
#include <iostream>
using namespace std;
void test04()
{
cout << "test04" << endl;
int c = 20;
int d = 30;
auto Add = [c, &d](int a, int b) {
cout << "c = " << c << endl;
cout << "d = " << d << endl;
return c;
};
d = 10;//在这里修改d的值,会改变Add里的d值吗?
cout << Add(1, 2) << endl;
}
int main(int argc,char **argv)
{
//test01();
//test02();
//test03();
test04();
return 0;
}
执行结果:
$ ./lambda_test
test04
c = 20
d = 10
20
手动书写捕获列表有时候是非常复杂的,这种机械性的工作可以交给编译器来处理,这时候可以在捕获列表中写一个 & 或 = 向编译器声明采用引用捕获或者值捕获。编译器会将外部变量全部捕获。
#include <iostream>
using namespace std;
void test05()
{
cout << "test05" << endl;
int c = 20;
int d = 30;
auto Add = [&](int a, int b) {
cout << "c = " << c << endl;
cout << "d = " << d << endl;
return c;
};
d = 10;//在这里修改d的值,会改变Add里的d值吗?
cout << Add(1, 2) << endl;
}
void test06()
{
cout << "test06" << endl;
int c = 20;
int d = 30;
auto Add = [=](int a, int b) {
cout << "c = " << c << endl;
cout << "d = " << d << endl;
return c;
};
d = 10;//在这里修改d的值,会改变Add里的d值吗?
cout << Add(1, 2) << endl;
}
int main(int argc,char **argv)
{
//test01();
//test02();
//test03();
test05();
test06();
return 0;
}
输出:
$ ./lambda_test
test05
c = 20
d = 10
20
test06
c = 20
d = 30
20
捕获列表’[]'中为空,表示Lambda不能使用所在函数中的变量。
void test07()
{
cout << "test07" << endl;
int c = 20;
int d = 30;
auto Add = [](int a, int b) {
cout << "c = " << c << endl; // 编译报错
cout << "d = " << d << endl; // 编译报错
return c; // 编译报错
};
d = 10;
cout << Add(1, 2) << endl;
}
编译报错:
lambda_test.cc:95:14: note: the lambda has no capture-default
auto Add = [](int a, int b) {
^
lambda_test.cc:93:6: note: ‘int c’ declared here
int c = 20;
^
lambda_test.cc:97:21: error: ‘d’ is not captured
cout << "d = " << d << endl;
上面提到的值捕获、引用捕获都是已经在外层作用域声明的变量,因此这些捕获方式捕获的均为左值,而不能捕获右值。
C++14之后支持捕获右值,允许捕获的成员用任意的表达式进行初始化,被声明的捕获变量类型会根据表达式进行判断,判断方式与使用 auto 本质上是相同的:
#include <iostream>
#include <memory>
using namespace std;
void test08()
{
cout << "test08" << endl;
auto important = make_unique<int>(1);
auto Add = [v1 = 1, v2 = std::move(important)](int a, int b)->int{
return a + b + v1 + (*v2);
};
cout << Add(1, 2) << endl;
}
int main(int argc,char **argv)
{
test08();
return 0;
}
执行结果:
$ ./lambda_test
test08
5
在C++14之前,lambda表示的形参只能指定具体的类型,没法泛型化。从 C++14 开始, Lambda 函数的形式参数可以使用 auto关键字来产生意义上的泛型。
简单点说,就是通过auto使lambda自适应参数类型:
#include <iostream>
using namespace std;
void test09()
{
cout << "test09" << endl;
auto Add = [](auto a, auto b) {
return a + b;
};
cout << Add(1, 2) << endl;
cout << Add(1.1, 2.2) << endl;
}
int main(int argc,char **argv)
{
test09();
return 0;
}
执行结果:
./lambda_test
test09
3
3.3
(1)采用值捕获的方式,lambda不能修改其值,如果想要修改,使用mutable修饰。
(2)采用引用捕获的方式,lambda可以直接修改其值。
#include <iostream>
using namespace std;
void test10()
{
cout << "test10" << endl;
int v = 10;
// 值捕获方式,使用mutable修饰,可以改变捕获的变量值
auto tes = [v]() mutable {
return ++v;
};
v = 5;
auto a = tes();// a=11;
cout << a << endl;
}
void test11()
{
cout << "test11" << endl;
int v = 10;
auto Add = [&v]{
return v++;
};
v = 6;
cout << Add() << endl;
}
int main(int argc,char **argv)
{
test10();
test11();
return 0;
}
执行结果:
$ ./lambda_test
test10
11
test11
6
#include <iostream>
using namespace std;
void test12()
{
cout << "test12" << endl;
int c = 12;
int d = 30;
int e = 30;
// auto Add = [&, d, e](int a, int b)
auto Add = [=, &c](int a, int b) -> int {
c = a;
cout << "d=" << d << ", e=" << e << endl;
return c;
};
d = 20;
cout << Add(1, 2) << endl;
cout << "c:" << c << endl;
}
int main(int argc,char **argv)
{
test12();
return 0;
}
测试结果:
$ ./lambda_test
test12
d=30, e=30
1
c:1
| 捕获 | 含义 |
|---|---|
| [] | 空捕获列表,Lambda不能使用所在函数中的变量。 |
| [names] | names是一个逗号分隔的名字列表,这些名字都是Lambda所在函数的局部变量。默认情况下,这些变量会被拷贝,然后按值传递,名字前面如果使用了&,则按引用传递 |
| [&] | 隐式捕获列表,Lambda体内使用的局部变量都按引用方式传递 |
| [=] | 隐式捕获列表,Lanbda体内使用的局部变量都按值传递 |
| [&,identifier_list] | identifier_list是一个逗号分隔的列表,包含0个或多个来自所在函数的变量,这些变量采用值捕获的方式,其他变量则被隐式捕获,采用引用方式传递,identifier_list中的名字前面不能使用&。 |
| [=,identifier_list] | identifier_list中的变量采用引用方式捕获,而被隐式捕获的变量都采用按值传递的方式捕获。identifier_list中的名字不能包含this,且这些名字面前必须使用&。 |
本专栏知识点是通过<零声教育>的系统学习,进行梳理总结写下文章,对c/c++linux系统提升感兴趣的读者,可以点击链接,详细查看详细的服务:C/C++服务器课程
假设我做了一个模块如下:m=Module.newdoclassCendend三个问题:除了对m的引用之外,还有什么方法可以访问C和m中的其他内容?我可以在创建匿名模块后为其命名吗(就像我输入“module...”一样)?如何在使用完匿名模块后将其删除,使其定义的常量不再存在? 最佳答案 三个答案:是的,使用ObjectSpace.此代码使c引用你的类(class)C不引用m:c=nilObjectSpace.each_object{|obj|c=objif(Class===objandobj.name=~/::C$/)}当然这取决于
我想在一个没有Sass引擎的类中使用Sass颜色函数。我已经在项目中使用了sassgem,所以我认为搭载会像以下一样简单:classRectangleincludeSass::Script::FunctionsdefcolorSass::Script::Color.new([0x82,0x39,0x06])enddefrender#hamlengineexecutedwithcontextofself#sothatwithintemlateicouldcall#%stop{offset:'0%',stop:{color:lighten(color)}}endend更新:参见上面的#re
我正在尝试用ruby中的gsub函数替换字符串中的某些单词,但有时效果很好,在某些情况下会出现此错误?这种格式有什么问题吗NoMethodError(undefinedmethod`gsub!'fornil:NilClass):模型.rbclassTest"replacethisID1",WAY=>"replacethisID2andID3",DELTA=>"replacethisID4"}end另一个模型.rbclassCheck 最佳答案 啊,我找到了!gsub!是一个非常奇怪的方法。首先,它替换了字符串,所以它实际上修改了
我有一些代码在几个不同的位置之一运行:作为具有调试输出的命令行工具,作为不接受任何输出的更大程序的一部分,以及在Rails环境中。有时我需要根据代码的位置对代码进行细微的更改,我意识到以下样式似乎可行:print"Testingnestedfunctionsdefined\n"CLI=trueifCLIdeftest_printprint"CommandLineVersion\n"endelsedeftest_printprint"ReleaseVersion\n"endendtest_print()这导致:TestingnestedfunctionsdefinedCommandLin
我没有理解以下行为(另请参阅inthisSOthread):defdef_testputs'def_test.in'yieldifblock_given?puts'def_test.out'enddef_testdoputs'def_testok'endblock_test=procdo|&block|puts'block_test.in'block.callifblockputs'block_test.out'endblock_test.calldoputs'block_test'endproc_test=procdoputs'proc_test.in'yieldifblock_gi
如何在Ruby中按名称传递函数?(我使用Ruby才几个小时,所以我还在想办法。)nums=[1,2,3,4]#Thisworks,butismoreverbosethanI'dlikenums.eachdo|i|putsiend#InJS,Icouldjustdosomethinglike:#nums.forEach(console.log)#InF#,itwouldbesomethinglike:#List.iternums(printf"%A")#InRuby,IwishIcoulddosomethinglike:nums.eachputs在Ruby中能不能做到类似的简洁?我可以只
说在前面这部分我本来是合为一篇来写的,因为目的是一样的,都是通过独立按键来控制LED闪灭本质上是起到开关的作用,即调用函数和中断函数。但是写一篇太累了,我还是决定分为两篇写,这篇是调用函数篇。在本篇中你主要看到这些东西!!!1.调用函数的方法(主要讲语法和格式)2.独立按键如何控制LED亮灭3.程序中的一些细节(软件消抖等)1.调用函数的方法思路还是比较清晰地,就是通过按下按键来控制LED闪灭,即每按下一次,LED取反一次。重要的是,把按键与LED联系在一起。我打算用K1来作为开关,看了一下开发板原理图,K1连接的是单片机的P31口,当按下K1时,P31是与GND相连的,也就是说,当我按下去时
我需要一个通过输入字符串进行计算的方法,像这样function="(a/b)*100"a=25b=50function.something>>50有什么方法吗? 最佳答案 您可以使用instance_eval:function="(a/b)*100"a=25.0b=50instance_evalfunction#=>50.0请注意,使用eval本质上是不安全的,尤其是当您使用外部输入时,因为它可能包含注入(inject)的恶意代码。另请注意,a设置为25.0而不是25,因为如果它是整数a/b将导致0(整数)。
我需要从json记录中获取一些值并像下面这样提取curr_json_doc['title']['genre'].map{|s|s['name']}.join(',')但对于某些记录,curr_json_doc['title']['genre']可以为空。所以我想对map和join()使用try函数。我试过如下curr_json_doc['title']['genre'].try(:map,{|s|s['name']}).try(:join,(','))但是没用。 最佳答案 你没有正确传递block。block被传递给参数括号外的方法
我经常将预配置的lambda插入可枚举的方法中,例如“map”、“select”等。但是“注入(inject)”的行为似乎有所不同。例如与mult4=lambda{|item|item*4}然后(5..10).map&mult4给我[20,24,28,32,36,40]但是,如果我制作一个2参数lambda用于像这样的注入(inject),multL=lambda{|product,n|product*n}我想说(5..10).inject(2)&multL因为“inject”有一个可选的单个初始值参数,但这给了我......irb(main):027:0>(5..10).inject