public class LambdaTest {
public static void main(String ... args){
int portNumber = 1337;
Runnable r = ()-> {
portNumber = 1338;
System.out.println(portNumber);
};
r.run();
}
}
如上代码,lambda里面要访问局部变量会报如照片错误:
在介绍为什么会报错的原因之前,稍作先介绍什么Lambda表达式
一、什么是lambda表达式
在java8实战这本书中,将Lambda表达式解释为:可传递的匿名函数的一种方式:它没有名称,但他有参数列表、函数主体、返回类型,可能还有一个可以抛出的异常列表
二、在哪里使用Lambda表达式
1.函数式接口:只定义了一个抽象方法的接口
2.函数描述符:函数式接口的抽象方法的签名基本上就是Lambda表达式的签名,这个抽象方法叫做函数描述符
三、函数式接口可以干什么?
lambda表达式可以以内联的形式为接口的抽象方法提供实现,并把整个表达式作为函数式接口的实例即创建一个函数式接口的实例
四、为什么lambda里面要访问局部变量必须是final
现在再来解释为什么lambda里面要访问局部变量必须是final?
首先:lambda表示可以无限制捕获实例变量(即表达式主体中的引用)和静态变量
但是,局部变量必须是显示声明为final或事实上是final
那么这lambda里面要赋值局部变量必须是final有什么关系?
实例变量:存储在堆中
局部变量:则保存在栈上
lambda表达式以内联的形式创建一个函数式接口的实例,保存在堆中,而局部变量则保存在栈中,可能造成实例对象得生命周期很有可能超过局部变量得生命周期:
1.局部变量声明周期:当该方法被调用时,该方法中的局部变量在栈中被创建。当方法调用结束时,退栈,这些局部变量全部死亡。而函数式接口实例对象生命周期和其他类对象一样:自创建一个实例对象,系统为该对象分配内存,直到没有引用变量指向分配给该对象得内存,它被JVM垃圾回收,所以完全可能出现的一种情况是:方法已调用结束,局部变量已死亡,但实例对象的对象仍然活着。
2.如果实例对象访问了同一个方法中得局部变量,就要求只要实例对象还活着,那么栈中的那些它要所访问的局部变量就不能死亡
因为创建一个函数式接口实例实际也是创建一个匿名内部类,这里是从匿名内部类的角度分析
以下是《java8实战》解释:
1.Lambda可以直接访问局部变量,而且Lambda是在一个线程中使用的,则使用Lambda的线程,可能会在分配该变量的线程将这个变量收回之后,去访问该变量。因此,Java在访问自由局部变量时,实际上是在访问它的副本,而不是访问原始变量。
2.这一限制不鼓励你使用改变外部变量的典型命令式编程模式
五、lambda里面要赋值局部变量而不使用final
则局部变量使用=》引用类型
属于个人理解,有不对的地方也请指出。