今天学习AQS时,发现在main方法中实例一个ReentrantLock,在线程内部类能够直接使用,当时就很奇怪,因为按照自己的理解,想要使用外部的变量,一般都是通过类的方法传递。
后来查了下资料,方法中的内部类是可以直接使用外部变量的。
但是有个前提就是该变量必须是final或者effectively final的。
effectively final就是指变量初始化后没有被更改。
对于基础数据类型,就是值不能被更改。
对于引用类型,就是指向不能更改。
也就是说我们可以在内部类直接使用lock,但是不能更改lock的引用指向。
main{
ReentrantLock lock =new ReentrantLock();
Runnable runnable = new Runnable() {
@Override
public void run() {
lock.lock();
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
};
Thread test = new Thread(runnable);
test.start();
lock=null;//出现编译错误
}
同理使用Integer 在内部类中 加减一样出错,因为缓存的原因。
后面又想到一个问题。
如果主线程结束了,还能使用lock吗?
在此之前,我已经使用了sout的方法分别在主线程和其他线程里打印了这个lock。发现是同一个lock。也就是指向相同。
主线程结束后,lock在主线程的栈中已经消失了,那么为什么另外一个线程还能使用呢?
后来查询资料,是第二个线程复制了一份引用在自己的线程栈中,这样就算主线程的引用销毁,但是子线程还能用。
本文介绍了Java中ReentrantLock的使用,特别是其在线程内部类中的应用。强调了外部变量在内部类中必须是final或effectively final的规则,并通过示例解释了即使主线程结束,子线程仍能访问共享锁的原因。讨论了线程间共享资源的生命周期及其对程序执行的影响。
641

被折叠的 条评论
为什么被折叠?



