Java中,成员变量再次赋值给局部变量,为什么这样写呢?

_斯人若彩虹 发布于 2020/06/12 15:13
阅读 964
收藏 0

比如 ThreadPoolExecutor 类中的成员变量:mainLock, 在方法中使用每次都被重新引用。这样写是出于什么样的考虑?

    private final ReentrantLock mainLock = new ReentrantLock();    


    private void interruptWorkers() {
        final ReentrantLock mainLock = this.mainLock;
        mainLock.lock();
        try {
            for (Worker w : workers)
                w.interruptIfStarted();
        } finally {
            mainLock.unlock();
        }
    }

 

加载中
2
Soutv
Soutv

大体来说,这样写是一种极端的优化,在运行时,即时编译器(JIT)将会把这种代码编译成与本地平台相关的机器码,并进行各种层次的优化,以提高性能。

http://mail.openjdk.java.net/pipermail/core-libs-dev/2010-May/004165.html

https://stackoverflow.com/questions/2785964/in-arrayblockingqueue-why-copy-final-member-field-into-local-final-variable

_斯人若彩虹
_斯人若彩虹
谢谢回答这个问题。
1
韦小仇
韦小仇

先说成员变量没有再次赋值给局部变量的情况(也就是我们通常的写法),此时对于final修饰的成员变量来说,JVM在执行成员方法时,由于语法要求final变量不可变,但JVM不能确定该final变量是否改变了,于是JVM每次运行成员方法时,都会从内存(而不是CPU缓存)中重新加载一次该变量。而对于题主提问的情况,将成员变量再次赋值给局部变量,指定JVM去使用赋值后的局部变量,无需理会成员变量,此时JVM就可以将该变量放在CPU缓存中,每次从CPU缓存中读取变量的值,不用去内存中重新加载,提高了运行速度。这是一种激进的优化策略,通过增加字节码来换取速度,相当于用空间换时间。而且作用其实并不大,除非某个方法频繁被执行、并且业务需求极其追求速度,否则并不建议在日常编程中这样写。

0
tcxu
tcxu

这么写,就是为了保证每次调用成员方法 private void interruptWorkers() 之时,使用的不变的(final)局部变量 mainLock, 就是 其实体对象的不变的(final)成员变量 mainLock。 

    对于一个final变量,如果是基本数据类型的变量,则其数值一旦被初始化,便不能更改。如果是引用类型的变量,如楼主所示,provate final ReentrantLock mainLock = new RecentrantLock(), 则在对其初始化之后便不能再让其指向另一个对象。
    在此案例的成员方法 private void interruptWorkers() 中, 首先声明的是一个 以 final 修饰的、该方法的局部引用  final ReentrantLock mainLock, 声明它的同时,立刻就用 该对象的 (this)、 已经在创建时就被初始化了的、不能再改变的引用 mainLock 来赋值:

final ReentrantLock mainLock = this.mainLock;

以此保证每次调用 interrupWorkers() 方法时所使用的、该方法的局部 (变量的) ReentrantLock 引用,始终如一。 就是说,方法的局部 final 变量  mainLock 的引用,始终是该对象的 final 的 成员变量 mainLock 的引用。
应指出,引用变量被final修饰之后,虽然不能再指向其他对象,但是它指向的对象的内容是可变的。

tcxu
tcxu
回复 @君君要上天哈 : 参看 Soutv 和 韦小仇 的回答。看看是否有道理。他们正在回答为什么。
君君要上天哈
君君要上天哈
回复 @tcxu : 这个我知道啊。但为什么要这样写?目的是啥?
tcxu
tcxu
回复 @君君要上天哈 : 成员方法 interrupWorkers() 里定义的 引用(final ReentrantLock mainLock),是只限于方法定义内的局部引用。每次调用,它使用的都是这个对象的 final(常量) 的引用的值:final ReentrantLock mainLock = this.mainLock。 就是说,使用这个对象的、不变的(final)引用。
君君要上天哈
君君要上天哈
没看懂
_斯人若彩虹
_斯人若彩虹
谢谢回答这个问题
0
网瘾少年徐志摩
网瘾少年徐志摩
该评论暂时无法显示,详情咨询 QQ 群:912889742
OSCHINA
登录后可查看更多优质内容
返回顶部
顶部