lambda表达式或者匿名函数中为什么要求外部变量为final

1、参考博客

  1. 关于Lambda表达式里面修改外部变量问题
  2. JDK8之前,匿名内部类访问的局部变量为什么必须要用final修饰

2、匿名内部类

在jdk7之前,匿名内部类访问外部类的局部变量时,那么这个局部变量必须用final修饰符修饰,如下图1所示。jdk8则不需要,但是我们在使用这个局部变量时,无法改变局部变量的值,否则编译会报错。
jdk7之前使用匿名函数
这个特点为什么要求是final类的呢?
因为在java设计之初为了保护数据的一致性而规定的。对引用变量来说是引用地址的一致性,对基本类型来说就是值的一致性。

注意:在JDK8之后,匿名内部类引用外部变量时虽然不用显式的用final修饰,但是这个外部变量必须和final一样,不能被修改(这是一个坑)。解决方案:可以通过定义一个相同类型的变量b,然后将该外部变量赋值给b,匿名内部类引用b就行了,然后就可以继续修改外部变量。

3、lambda表达式

这是因为:Java会将result的值作为参数传递给Lambda表达式,为Lambda表达式建立一个副本,它的代码访问的是这个副本,而不是外部声明result变量。可能很多同学会问为什么非要建立副本呢,直接访问外部的result变量得多方便呢。答案是:这是不可能滴,因为result定义在栈中,当Lambda表达式被执行的时候,result可能已经被释放掉了。

当然啦,你要是一定要在Lambda表达式里面修改外部变量的值也是可以的,可以将变量
定义为实例变量或者将变量定义为数组。

① 情形一

如下图所示:在使用外部类的局部变量时,如果试图修改值,就会编译报错。
在使用lambda表达式

② 情形二

如果局部变量是对象类型,则对象的引用地址不可改变,如下图所示。但是如果在局部变量中修改对象是没有问题的(第二篇博客有详细解释)。
在这里插入图片描述

③ 情形三

在项目中,我遇到过这种情况:有两个集合,我先对一个集合进行遍历,在这里面又需要对另外一个集合遍历。当时出现了编译报错:variable used in lambda expression should be final or effectively final(lambda表达式中使用的变量应该是final或effective final),位置在下图的红框内。如下图所示。但是下图当我工作之余究其缘由调试时的demo,此时编译又不报错了。我能力有限,暂时还不清楚问题出现的原因。当时我的解决方案有两个:一个是在将lambda表达式换成了for下标遍历,就OK了。另一种方案是创建一个报错的同类型新对象,将list1的引用赋值给新对象,编译也不报错了(如下下图所示)。
在这里插入图片描述
在这里插入图片描述

为什么lambda表达式使用的局部变量要是final的

为什么 Lambda 表达式(匿名类) 不能访问非 final 类型的局部变量呢?

之前我一直认为是由于Lambda 表达式(匿名类) 会在另一个线程中执行,实例变量存在堆中,而局部变量是在栈上分配,如果在线程中要直接访问一个局部变量,可能线程执行时该局部变量已经被销毁了。

但是有人提出这个原因存在问题,说他们是在同一个线程中运行的,大兄弟提供的验证代码如下:

public class Java8Tester {public static void main(String[] args) {String str = "Hello, ";Person person = name -> {System.out.println(str + name + "   " + Thread.currentThread().getId());System.out.println("匿名类" + "   " + Thread.currentThread().getId());};System.out.println("主方法" + "   " + Thread.currentThread().getId());person.say("JC");}interface Person {void say(String name);}
}

运行之后的结果显示确实都是main线程运行的,结果如下:
在这里插入图片描述
我又去看了一些其他的博客,有人说是Java为了防止数据不同步而规定的,也就是为了防止在lambda表达式内使用的外层局部变量被外层代码修改之后内部无法同步这个修改。想看原贴可以点我

我觉得上面超链接中讲述的原因是比较好的,如果其他同学有不同意见欢迎一起交流一起进步。

Java 8 的 Lambda 可以捕获什么变量呢?

(1). 捕获实例变量或静态变量是没有限制的 (可认为是通过 final 类型的局部变量 this 来引用前两者);

(2). 捕获的局部变量必须显式的声明为 final 或实际效果的的 final 类型
注意(敲黑板):如果在Lambda表达式中使用局部变量,即使我们没有声明成final类型的,编译器也会帮助我们将他们声明成final的,所以如果重新赋值会出错。

如下:
在这里插入图片描述
结果如下:
在这里插入图片描述
此时程序不会报错,虽然我们没有使用final修饰变量b,但是Lambda表达式中也能使用,这是因为编译器帮我们自动声明成final的。

众所周知,final类型的变量不能重新赋值,来验证下是不是编译器真的帮我们声明成了final,如下:
在这里插入图片描述
报错:
在这里插入图片描述
那么是什么时候帮我们声明成final呢?答案是在创建变量的时候直接声明成final的,测试如下:
在这里插入图片描述
在这里插入图片描述

总结:在Lambda表达式中可以捕获静态变量和实例变量,但是如果想要捕获局部变量的时候就需要声明成final的,即使我们不主动声明,编译器也会为我们自动声明成final的,不能再重新赋值。也就是Lambda表达式中访问的局部变量(隐式被声明为final的)是可读不可写的,但是Lambda表达式中访问的实例变量和静态变量是可读可写的。




在这里插入图片描述

为什么会这样?

在Java中lambda表达式是匿名类语法上的进一步简化,它的本质其实还是调用对象的方法。lambda表达式会以内联的形式创建一个函数式接口的实例,保存在堆中,而局部变量则保存在栈中,而在Java中方法调用是值传递的(特别声明java中都是按值传递的!!!),所以在lambda表达式中对变量的操作都是基于原变量的副本,不会影响到原变量的值。那假如没有要求lambda表达式外部变量为final修饰,那么就会误以为外部变量的值能够在lambda表达式中被改变,而这实际是不可能的,所以要求外部变量为final。
我们知道局部变量随着方法的调用会被压入栈中,当方法调用结束时,出栈,这些局部变量全部死亡。而函数式接口实例对象生命周期和其他类对象是一样的,从创建一个实例对象开始,系统就会为该对象分配内存,直到没有引用变量指向分配给该对象得内存,它被GC垃圾回收,所以很可能出现的一种情况就是:方法已调用结束,局部变量已死亡,但实例对象的对象仍然活着。也就是说如果这个时候仍然有引诱变量指向lambda表达式在堆中所创建的实例,但是lambda表达式中的局部变量已经死亡,岂不是出了问题,这在java中是不允许的,在我还需要你的时候你的局部变量是不可以死亡的。

以上就是我对这个点的理解,小伙伴们有什么不同的见解或者文章中有什么错误的地方,欢迎大家在里评论区留言

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

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

相关文章

location 和 history

Location 对象:封装了浏览器地址栏的 URL 信息 一、hash 返回 URL 中 hash(#后跟零个或者多个字符), 如果不包含, 返回空字符串 # 位置标识符 : 当前页面的位置信息, 比如: 跳转顶部 console.log(location.hash); // ""二、host : 返回服务器名称和端口…

ble芯片 全称_蓝牙芯片都有哪些厂商?一文解答

蓝牙5新标准是蓝牙技术自1999年诞生以来推出的第十个标准版本。其性能上大幅提升,可归结为:更快、更长、更给力,非常适合运用于无线可穿戴、工业和智能家居等领域。但蓝牙技术产品能否真正进人批量生产在于芯片制造技术能否跟得上&#xff0c…

你知道Java中final和static修饰的变量是在什么时候赋值的吗?

开始 一位朋友在群里问了这样一个问题: 本着乐于助人的想法,我当时给出的回答: 后来我总觉得哪里不对劲,仔细翻阅了《Java虚拟机规范》和《深入理解Java虚拟机》这一部分的内容,害!发现自己理解的有问题。…

获取元素大小和位置的方式

一、直接获取元素样式属性值 – element.style.width console.log(div.style.width); // 500px console.log(parseInt(div.style.width)); // 500 console.log(typeof (div.style.width)); // string二、Offset 偏移量 offsetWidth width padding border offsetHeight he…

mybatis 取查询值_MyBatis-SELECT基本查询

1、返回一个LISTselect * from tbl_employee where last_name like #{lastName}2、将查询记录封装为一个Mapselect * from tbl_employee where id#{id}返回一条记录的map;key就是列名,值就是对应的值。3、多条记录封装为一个mapMapKey("id")pu…

ES6 里面的 class

ES5 对象的写法 let x 10,y 20;const obj {x: x,y: y,sum: function () {return this.x this.y;} };ES6 对象的写法 const obj {x,y,sum() {return this.x this.y;} };class 有点类似 java 的 class class Person { // 类名大写// 私有属性和方法 写在 construtor 里面…

normalize函数_Pandas 数据处理(一) —— 几个简单函数掌握!

对于 Pandas, 接触过 Python 数据处理的小伙伴们都应该挺熟悉的,做数据处理不可或缺的一个程序包,最大的特点高效,本篇文章将通过案例介绍一下 Pandas 的一些基础使用!1,读入数据大部分数据都可以用 read_c…

Java Collections.emptyList() 方法的使用及注意事项

Java Collections.emptyList方法的使用及注意事项 一、emptyList() 作用:返回一个空的List(使用前提是不会再对返回的list进行增加和删除操作);好处: 1. new ArrayList()创建时有初始大小,占用内存&#…

git 生成多个patch_详解如何使用git 生成patch 和打入patch

平时我们在使用git 管理项目的时候,会遇到这样一种情况,那就是客户使用git 生成patch 给到我们,那我们就需要把客户给到patch 打入到我们的project ,基于这样一个场景,我把git 如何生成patch 和如何打入patch 做总结生…

Java集合工具类的一些坑,Arrays.asList()、Collection.toArray()...

Arrays.asList() 使用指南 最近使用Arrays.asList()遇到了一些坑,然后在网上看到这篇文章:Java Array to List Examples 感觉挺不错的,但是还不是特别全面。所以,自己对于这块小知识点进行了简单的总结。 简介 Arrays.asList()…

python saltstack web_saltstack学习-8:web管理页面(halite)

安装halite方法一:失败,待查原因1、安装并启动salt-api yuminstall salt-api –yservice salt-api startchkconfig salt-api on2、安装halite及其依赖文件 yuminstall python-pip –ypipinstall --upgrade pippipinstall -U halitepipinstall cherrypypi…

什么是 NIO? NIO 和 BIO、AIO 之间的区别是什么?NIO主要用来解决什么问题?

1 BIO,NIO,AIO都有什么区别,NIO的原理是什么? BIO BIO:传统的网络通讯模型,就是BIO,同步阻塞IO, 其实就是服务端创建一个ServerSocket, 然后就是客户端用一个Socket去连…

关于页面配色

一、互补色 当两个颜色恰好在色环的两端时,这两个颜色叫做补色。补色搭配能形成强列的对比效果 在线配色工具地址 文字背景色和文字颜色互为补色,文字会很难看清,那么就只使用一种颜色作为主要颜色,其补色用来装点页面 比如&…

axure命令行_Axure变量详解

以前使用Axure只是停留在元件的布局和简单交互事件的设置,使用得非常肤浅,直到现在有时间静下心来重学Axure,才发现函数和变量的牛逼之处。以前在做较复杂交互时,为了实现一个效果吭哧吭哧写了一串命令,而往往函数和变…

一次性说清楚秒验(本机号码一键登录)基本原理、优势、场景、交互过程和常见的问题

一、 关于秒验(一键登录)基本原理 秒验(一键登录)产品整合了三大运营商特有的数据网关认证能力,升级短信验证码体验,应用于用户注册、登陆、支付、安全校验等场景,可实现用户无感知校验&#x…

php 基本语法

一、php 基本语法 <?php // echo | print 两种输出语句 echo 任何位置; print 任何位置; ?>二、注释 <?php // echo 任何位置; // 单行注释 /*多行注释*/ ?>三、变量声明 变量名以 $ 开始&#xff0c;后面跟变量的名字&#xff0c;区分大小写 <?php $x …

爬虫python能做游戏吗_一入爬虫深似海,从此游戏是路人!总结我的python爬虫学习笔记!...

1、基本抓取网页get方法post方法2、使用代理IP在开发爬虫过程中经常会遇到IP被封掉的情况&#xff0c;这时就需要用到代理IP&#xff1b;在urllib2包中有ProxyHandler类&#xff0c;通过此类可以设置代理访问网页&#xff0c;如下代码片段&#xff1a;3、Cookies处理cookies是某…

三大运营商实现本机号码一键登录原理与应用

很多APP的目前都支持“本机号码一键登录”功能。本机号码一键登录是基于运营商独有网关认证能力推出的账号认证产品。用户只需一键授权&#xff0c;即可实现以本机号码注册/登录&#xff0c;相比先前的短信验证码流程体验更优。 目前市面上有很多厂商提供三网验证的服务&#…

php 数据类型

七种数据类型&#xff1a; String / Integer / Float / Boolean / Array / Object / null 检测变量的数据类型和值 var_dump() 一、字符串 $x abc; var_dump($x); echo $x; // string abc二、整形 $x 1234; var_dump($x); echo $x; // int 1234三、浮点型 $x 1.1; var_d…

部署到gcp_剖析大数据公司为什么选择 GCP?

文章来源&#xff1a;加米谷大数据假如L 是一家大数据公司。下面我们的文章将围绕L展开分析。很多公司拥有大数据。每天早餐之前&#xff0c;健壮的日志框架就已经生成了 PB 级别的日志&#xff0c;并以防万一将这些数据长期保存在了亚马逊的 S3 上。还有一些公司会使用他们自己…