今天学习AQS时,发现在main方法中实例一个ReentrantLock,在线程内部类能够直接使用,当时就很奇怪,因为按照自己的理解,想要使用外部的变量,一般都是通过类的方法传递。
后来查了下资料,方法中的内部类是可以直接使用外部变量的。
但是有个前提就是该变量必须是final或者effectively final的。
effectively final就是指变量初始化后没有被更改。
对于基础数据类型,就是值不能被更改。
对于引用类型,就是指向不能更改。
也就是说我们可以在内部类直接使用lock,但是不能更改lock的引用指向。
main{ReentrantLock lock =new ReentrantLock();Runnable runnable = new Runnable() {@Overridepublic 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在主线程的栈中已经消失了,那么为什么另外一个线程还能使用呢?
后来查询资料,是第二个线程复制了一份引用在自己的线程栈中,这样就算主线程的引用销毁,但是子线程还能用。