C++ 中 Lambda 对变量的捕获居然是在声明时就做了

我看到在 C++ 14 特性中支持 Lambda 表达式捕获 move-only 的类型(原先是引用或值传递),于是写了一个小 demo:

auto p = std::make_unique<int>(1);
auto task1 = [p = move(p)](){ *p = 5; };
cout << *p << endl;
上述代码可以通过编译,但是运行时出现 core dump ,明显是因为此时 p 已经被 move 到 Lambda 表达式中去了,这里就迷惑,我一直以为要 auto task = [](){…}(); 或是 task1(); 执行时才会做参数的初始化(类似函数那样),但从结果来看,从声明的那一刻变量就已经被捕获了。

虽然这一点对原本的值传递和引用捕获的参数来讲没有什么感知,但是对于捕获 move-only 类型的函数确实会产生一定的影响,因为从 Lambda 声明的这一刻开始你原本的外部变量就不能使用了。

另外被捕获到 Lambda 内部的 move-only 变量是以类似 static 的状态存在的,即多次调用 task 对 *p 的改变会累积,以这个 demo 可以看的比较清楚:

auto p = std::make_unique<int>(1);
auto task1 = [p = move(p)](){
(*p)++;
cout << *p << endl;
};
task1();
task1();
task1();
task1 lambda auto 捕获6 条回复 • 2021-07-16 13:50:11 +08:00
wutiantong 1
wutiantong 1 天前 ❤️ 1
lambda 就是这样的,
你定义它时,你实际上是创建了一个匿名类型的 functor object,
而捕获列表中定义的都是这个匿名类型的成员变量,
所以,并不是 static 的,而是成员变量的。
nightwitch 2
nightwitch 1 天前 ❤️ 1
其实仔细想想就会发现你的思路不合理。
比如考虑一个 taskQueue,一个 lambda 会被推到这个队列里,而可能要等前面的任务都执行完毕以后才会执行。
如果 lambda 不在声明的时候拷贝 /移动变量,那么当 taskQueue 排到的时候可能这个变量生命周期早结束了。

lambda 其实是一个语法糖,本质上是声明一个重载了 operator()的 struct,auto task1 = []()… 类似于实例化这个结构体。
ipwx 3
ipwx 1 天前
楼主可能和 JS 搞起来了。但是 JS 那种其实不是更奇怪吗。

for (int i=0; i<10; ++i) {
task_list.emplace([i] () { std::cout << i << std::endl; });
}

这种用法不挺符合常识吗?
hitmanx 4
hitmanx 1 天前
“.. 但从结果来看,从声明的那一刻变量就已经被捕获了。虽然这一点对原本的值传递和引用捕获的参数来讲没有什么感知”..

Really?以值 capture 一样有感知

int main()
{
….int a = 1;
….auto func = [a]() { std::cout << a << std::endl; };
….a = 2;
….func();
….return 0;
}
mer 5
mer 1 天前
@wutiantong 原来是这样
@nightwitch 懂了
@ipwx 没用过 JS
@hitmanx 噢 我说的值传递不是这个意思,忘了考虑这种值 Capture 的,一般用 引用 capture 和,值传参
aneostart173 6
aneostart173 43 分钟前
跟 rust 一样啦?

Android Studio下配置和使用Lambda

和朋友讨论 JAVA8 的新特性,聊到Lambda,正好在掘金上看到一篇相关的文章,结合资料,作一个总结,特别是记录下实际使用中遇到的问题。

什么是Lambda表达式

lambda表达式,它将允许我们将行为传到函数里。在Java 8之前,如果想将行为传入函数,仅有的选择就是匿名类,需要6行代码。而定义行为*重要的那行代码,却混在中间不够突出。Lambda表达式取代了匿名类,取消了模板,允许用函数式风格编写代码。这样有时可读性更好,表达更清晰。

— Java8 lambda表达式10个示例

阅读完上面的文字估计也不是特别明白,对于我们日常开发Android,就是简化了匿名函数的使用,可以简单通过下面的示例来感受一下,如果你有更深的兴趣,文末有更多搜集的资料供你阅读。

Lamdba示例

用lambda表达式实现Runnable

使用lambda表达式可以替换匿名类,而实现Runnable接口是匿名类的*好示例。Java 8之前的runnable实现方法,需要4行代码,而使用lambda表达式只需要一行代码。只需要用() -> {}代码块替代整个匿名类。

  1. // Java 8之前:
  2. new Thread(new Runnable() {
  3. @Override
  4. public void run() {
  5. System.out.println(“在Java8之前, 需要写很多代码”);
  6. }
  7. }).start();
  8. //Java 8方式:
  9. new Thread( () -> System.out.println(“使用Java8, Lambda表达式一目了然”) ).start();

 

输出:

  1. 在Java8之前, 需要写很多代码
  2. 使用Java8, Lambda表达式一目了然

 

这个例子展示了Java 8 lambda表达式的语法,可以使用lambda写出如下代码:

  1. (params) -> expression
  2. (params) -> statement
  3. (params) -> { statements }

 

例如,如果你的方法不对参数进行修改、重写,只是在控制台打印点东西的话,那么可以这样写:

() -> System.out.println("Hello World");
  • 1

如果你的方法接收两个参数,那么可以写成如下这样:

(int a, int b) -> a + b
  • 1

使用lambda表达式进行事件处理

在Android日常开发中,我们常常会设置各种事件,比如setOnClickListenersetOnItemClickListener等等,下面对比下前后的写法变化:

  1. //之前
  2. viewA.setOnClickListener(new View.OnClickListener() {
  3. @Override
  4. public void onClick(View v) {
  5. //Do something
  6. }
  7. });
  8. //使用lambda
  9. viewA.setOnClickListener(v -> {
  10. //Do something
  11. });
  12. //或者
  13. viewA.setOnClickListener(View v -> {
  14. //Do something
  15. });

 

这样一对比是不是简洁很多?那么对于多个参数的setOnItemClickListener怎么写呢?

  1. //之前
  2. xxxListView.setOnItemClickListener(new AdapterView.OnItemClickListener() {
  3. @Override
  4. public void onItemClick(AdapterView<?> parent, View view, int position, long id) {
  5. //Do something
  6. }
  7. });
  8. //使用lambda后
  9. xxxListView.setOnItemClickListener((parent,view,position,id)->{
  10. //Do something
  11. });
  12. //或者
  13. xxxListView.setOnItemClickListener((AdapterView<?> parent, View view, int position, long id)->{
  14. //Do something
  15. });
  16. //甚至
  17. xxxListView.setOnItemClickListener((a,b,c,d)->{
  18. //Do something
  19. });

 

以上两种例子大概是日常Android开发中*为常用的,更多示例请访问 :Java8 lambda表达式10个示例

启用Lambda

启用Lamdba目前有两种方式,一个是使用Google官方的,一个是使用第三方Java8兼容插件,推荐使用第三方兼容插件 。

基本要求如下:

  • Android Studio 2.1+
  • 安装好 JAVA 8

方式一:Google官方方式

要使用新的 Java 8 语言功能,还需使用新的 Jack 工具链。新的 Android 工具链将 Java 源语言编译成 Android 可读取的 Dalvik 可执行文件字节码,且有其自己的 .jack 库格式,在一个工具中提供了大多数工具链功能:重新打包、压缩、模糊化以及 Dalvik 可执行文件分包。

以下是构建 Android Dalvik 可执行文件可用的两种工具链的对比:

旧版 javac 工具链:
javac (.java --> .class) --> dx (.class --> .dex)
新版 Jack 工具链:
Jack (.java --> .jack --> .dex)
配置 Gradle
如需为您的项目启用 Java 8 语言功能和 Jack,请在模块层级的 build.gradle 文件中输入以下内容:

  1. android {
  2. defaultConfig {
  3. jackOptions {
  4. enabled true
  5. }
  6. }
  7. compileOptions {
  8. sourceCompatibility JavaVersion.VERSION_1_8
  9. targetCompatibility JavaVersion.VERSION_1_8
  10. }
  11. }

 

已知问题
Instant Run 目前不能用于 Jack,在使用新的工具链时将被禁用。

Java 8 语言功能

方式二:使用第三方Java8兼容插件

下面是插件的ReadMe的配置:添加下面的内容到项目的build.gradle文件中

  1. buildscript {
  2. repositories {
  3. mavenCentral()
  4. }
  5. dependencies {
  6. classpath 'me.tatarka:gradle-retrolambda:3.2.5'
  7. }
  8. }
  9. // Required because retrolambda is on maven central
  10. repositories {
  11. mavenCentral()
  12. }
  13. apply plugin: 'com.android.application' //or apply plugin: 'java'
  14. apply plugin: 'me.tatarka.retrolambda'

在本人的项目中,是如下配置,可以避免很多不必要的错误:

build.gradle文件在项目根目录有一个,在Module下也有一个:

/build.gradle

  1. buildscript {
  2. repositories {
  3. jcenter()
  4. mavenCentral()
  5. }
  6. dependencies {
  7. classpath ‘com.android.tools.build:gradle:2.1.2’
  8. classpath ‘me.tatarka:gradle-retrolambda:3.2.5’
  9. classpath ‘me.tatarka.retrolambda.projectlombok:lombok.ast:0.2.3.a2’
  10. }
  11. configurations.classpath.exclude group: ‘com.android.tools.external.lombok’
  12. }
  13. allprojects {
  14. repositories {
  15. jcenter()
  16. mavenCentral()
  17. }
  18. }

 

/app/build.gradle

  1. apply plugin: ‘com.android.application’
  2. apply plugin: ‘me.tatarka.retrolambda’
  3. android {
  4. compileOptions {
  5. sourceCompatibility JavaVersion.VERSION_1_8
  6. targetCompatibility JavaVersion.VERSION_1_8
  7. }
  8. }

 

可以看到我在使用中多了几个配置,不过都是在这个插件的issue里查到的。建议仔细阅读下配置说明 ,可以适应一些实际项目中的一些特别需求。

常见Error及解决方式

Couldnot find property ‘options’ on task ‘:app:compileDebugJavaWithJack’.

原因:使用Google官方的方式有一定的兼容性,使用Jack时不能同时使用APT,如果使用butterknife、Dagger等使用了APT的注解框架就不行了

解决方案:使用retrolambda的兼容插件的方式启用lambda

Doesn’t seem to correctly use modified lombok version

原因:参见这里

解决方案:按照我上面写的就不会出现这个问题了。

总结

对于想尝鲜的的开发者,启用Lamdba是个不错的选择,毕竟这个东西会慢慢普及的。

AWS API Gateway和Lambda的简单测试

文章目录
具体流程
创建Lambda应用
创建API Gateway
S3 Bucket网页测试触发API Gateway和Lambda函数
碰到S3 Bucket删除问题

测试了一下使用API Gateway来触发Lambda应用,并将结果显示在网页上。
Lambda支持语言为: Node.js,Java,Python,C#,Go,PowerShell,等等。它具有无需管理服务器,根据需求自动扩展,便宜等优势。
具体流程
在S3 Bucket里存放网页,显示网页后,在网页里通过API Gateway来触发Lambda功能应用,并将Lambda功能应用返回的结果显示在网页上。Route53来使用DNS网址为可选。

创建Lambda应用
Lambda是EC2在一起的Compute界面下,点进去后,再点Create Function来创建函数, 取名为:XiongLambdaFunction,语言设置为Python 3.6,这样会自动生成以下代码:
%title插图%num

进行以下替换:

%title插图%num

创建API Gateway
创建一个API Gateway,在Networking & Content Delivery下点API Gateway,选择
REST,API取名为:XiongTestAPI,在Actions下点Create Method,选择GET方法,Integration type选择Lambda Function,并选择前面已经创建的XiongLambdaFunction。点保存,有以下提示。

You are about to give API Gateway permission to invoke your Lambda function:
arn:aws:lambda:ap-southeast-1:606255748358:function:XiongLambdaFunction

点Deploy API对API进行发布,发布成功后会有一个URL生成,这个生成的URL将会被用做第三步测试:

Invoke URL: https://p9ciqi7imh.execute-api.ap-southeast-1.amazonaws.com/XiongReturnNameAPI

S3 Bucket网页测试触发API Gateway和Lambda函数
创建两个文件:Index.html和error.html,此测试是根据ACloud Guru所做的示例。
在index.html里,将GET请求的URL替换为: https://p9ciqi7imh.execute-api.ap-southeast-1.amazonaws.com/XiongReturnNameAPI,如下html代码所示:
%title插图%num

error.html

%title插图%num

创建一个S3 Bucket,取名为: xionglambdatest,在Properties下面进行设置,Static website hosting,创建了以下Endpoint, Index document设置为index.html,Error Document设置为error.html,保存。
Static website hosting
Endpoint : http://xionglambdatest.s3-website-ap-southeast-1.amazonaws.com

将前面创建的index.html和error.html进行上传,

如果你注册了域名的话,可以使用Route53来设置 A Record来指向S3 Bucket的Endpoint达到域名访问的目的。

这里直接测试:
Static website hosting
Endpoint : http://xionglambdatest.s3-website-ap-southeast-1.amazonaws.com

返回以下错误:

%title插图%num

在S3 Bucket xionglambdatest中,设置Make Public,确认,再试,可以成功访问。

点按钮无反应,在Chrome的Develop Tool里发现碰到以下错误:

%title插图%num

CORS在作怪,在Resource下里的Action里点Enable CORS,再试, 点按钮后,界面返回如下:

%title插图%num

说明https://p9ciqi7imh.execute-api.ap-southeast-1.amazonaws.com/XiongReturnNameAPI返回了{“statusCode”: 200, “headers”: {“Access-Control-Allow-Origin”: “*”}, “body”: “Xiong Huilin”}。

重新再试,在创建GET方法时,记得把Use Lambda Proxy integration选择项选上, 表明是Lambda返回值 。另外记得在Resource下里的Action里点Enable CORS, *后Deploy API。这次成功。

点按钮后,以下文字:
Hello 2nd December 2019!
变为:
Hello Xiong Huilin。
也就是https://p9ciqi7imh.execute-api.ap-southeast-1.amazonaws.com/XiongReturnNameAPI返回了Xiong Huilin,测试成功!

点完按钮的效果如下:
%title插图%num

碰到S3 Bucket删除问题
前面做Elastic Beanstalk测试的时候:AWS的CloudFormation和Elastic Beanstalk的简单测试,自动创建了名为elasticbeanstalk-ap-southeast-1-606255748358的S3 Bucket,删除时返回了Access Denied的错误。

参见: AWS Forum: can’t delete empty elasticbeanstalk created bucket

在以下Bucket Policy的JSON文件里将Deny改成Allow,就可以删除了,问题解决。
%title插图%num

参见Bucket Policy Examples,可以使用以下JSON文件设置Bucket Policy以达到Bucket为公共访问的目的:

%title插图%num