匿名类 java final_java为什么匿名内部类的参数引用时final(转)

https://blog.csdn.net/z69183787/article/details/68490440

喜欢看生肉的同学就不用看我的回答了,直接看R大的三篇回答,尤其是第一篇后面的回复部分。

我只是试着用大白话做个简单的整理,希望能更容易理解一点。

关于对象与闭包的关系的一个有趣小故事

(这篇的精华在后面的回复,小故事可以跳过)

1.

闭包(Closure)

什么是闭包,大白话不怎么严谨的说就是:

一个依赖于外部环境自由变量的函数

这个函数能够访问外部环境里的自由变量

看下面这个Javascript闭包的例子:

d8ba52585120817d8f0011a25a701b3b.png

对内部函数function(x)来讲,y就是自由变量,而且function(x)的返回值,依赖于这个外部自由变量

y。而往上推一层,外围Add(y)函数正好就是那个包含自由变量y的环境。而且Javascript的语法允

许内部函数function(x)访问外部函数Add(y)的局部变量。满足这三个条件,所以这个时候,外部函

数Add(y)对内部函数function(x)构成了闭包。

闭包的结构,如果用λ演算表达式来写,就是多参数的Currying技术。

>λx.λy.x+y

但在Java中我们看不到这样的结构。因为Java主流语法不允许这样的直接的函数套嵌和跨域访问变量。

2.

类和对象

但Java中真的不存在闭包吗?正好相反,Java到处都是闭包,所以反而我们感觉不出来在使用闭

包。因为Java的“对象”其实就是一个闭包。其实无论是闭包也好,对象也好,都是一种数据封装的

手段。看下面这个类

434821d682ebcbde9c8a22f410a81d68.png

看上去x在函数add()的作用域外面,但是通过Add类实例化的过程,变量”x“和数值”2“之间已经绑

定了,而且和函数add()也已经打包在一起。add()函数其实是透过this关键字来访问对象的成员字

段的。

如果对闭包有疑问,可以看这个更详细的回答:

3.

Java内部类是闭包:包含指向外部类的指针

那Java里有没有除了实例对象之外的闭包结构?Java中的内部类就是一个典型的闭包结构。例子如下,

3eaccf061fae7e81f721650f119c8b17.png

下图画的就是上面代码的结构。内部类(Inner

Class)通过包含一个指向外部类的引用,做到自

由访问外部环境类的所有字段,变相把环境中的自由变量封装到函数里,形成一个闭包。

1f9fd2e1b018fe29b25a869e1da4cc68.png

4.

别扭的匿名内部类

但Java匿名内部类就做得比较尴尬。下面这个例子中,getAnnoInner负责返回一个匿名内部类的引用。

dd74284ca16365ca52d221e40260e69c.png

匿名内部类因为是匿名,所以不能显式地声明构造函数,也不能往构造函数里传参数。不但返回的只是个叫AnnoInner的接口,而且还没有和它外围环境getAnnoInner()方法的局部变量x和y构成任何类的结构。但它的addXYZ()函数却直接使用了x和y这两个自由变量来计算结果。这就说明,外部方法getAnnoInner()事实上已经对内部类AnnoInner构成了一个闭包。

但这里别扭的地方是这两个x和y都必须用final修饰,不可以修改。如果用一个changeY()函数试图修改外部getAnnoInner()函数的成员变量y,编译器通不过,

error:

cannot assign a value to final variable y

这是为什么呢?因为这里Java编译器支持了闭包,但支持地不完整。说支持了闭包,是因为编译器编译的时候其实悄悄对函数做了手脚,偷偷把外部环境方法的x和y局部变量,拷贝了一份到匿名内部类里。如下面的代码所示。

218f8c81841c483ba9ff1fdcf53c6997.png

所以用R大回答里的原话说就是:

Java编译器实现的只是capture-by-value,并没有实现capture-by-reference。

而只有后者才能保持匿名内部类和外部环境局部变量保持同步。

但Java又不肯明说,只能粗暴地一刀切,就说既然内外不能同步,那就不许大家改外围的局部变量。

5.

其他和匿名内部类相似的结构

《Think

in Java》书里,只点出了匿名内部类来自外部闭包环境的自由变量必须是final的。但实际上,其他几种不太常用的内部类形式,也都有这个特性。

比如在外部类成员方法内部的内部类。

f22961a83cab66363d9db6f7193c4d01.png

比如在一个代码块block里的内部类。

aa2d02c92d863dd6be0e68678035835d.png

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值