final 是Java 中重要关键字之一,能够应用于类、方法以及变量上。这篇文章中将讲解什么是 final 关键字?将变量、方法和类声明为 final 表明了什么?使用 final 的好处是什么?java
final 在 Java 中是一个保留的关键字,能够声明成员变量、方法、类以及本地变量。一旦你将引用声明做 final,你将不能改变这个引用了,编译器会检查代码,若是试图将变量再次初始化的话,编译器会报编译错误。编程
凡是对成员变量或者本地变量(在方法中的或者代码块中的变量称为本地变量)声明为 final 的都叫做 final 变量。final 变量常常和 static 关键字一块儿使用,做为常量。下面是 final 变量的例子:缓存
public static final String NAME = "wupx"; NAME = new String("wupx"); //invalid compilation error
final 变量是只读的。安全
final 也能够声明方法,Java 里用 final 修饰符去修饰一个方法的惟一正确用途就是表达:这个方法本来是一个虚方法,如今经过 final 来声明这个方法不容许在派生类中进一步被覆写(override)。多线程
Java 中非私有的成员方法默认都是虚方法,而虚方法就能够在派生类中被覆写。为了保证某个类上的某个虚方法不在派生类中被进一步覆写,就须要使用 final 修饰符来声明,让编译器(例如 javac)与 JVM 共同检查并保证这个限制老是成立。ide
下面引用 R 大 在知乎上的回答来打破“用 final 修饰方法可让对这个方法的调用变快”的流言:函数
曾经有一种广为流传的说法是用final修饰方法可让对这个方法的调用变快。这种说法在现代主流的优化JVM上都是不成立的(例如Oracle JDK / OpenJDK HotSpot VM、IBM J9 VM、Azul Systems Zing VM等)。这是由于:能用final修饰的虚方法,其派生类中必然不可能存在对其覆写的版本,因而能够断定这个虚方法只有一个可能的调用目标;而若是此时把这个final修饰符去掉,这些先进的JVM均可以经过“类层次分析”(Class Hierarchy Analysis,CHA)来发现这个方法在派生类中没有进一步覆写的版本,因而一样能够断定这个虚方法只有一个可能的调用目标。而后二者的优化程度会如出一辙,不管是从“不须要经过虚分派而能够直接调用目标”(称为“去虚化”,devirtualization)仍是从“便于内联”的角度看,这两种状况都是同样的。性能
而若是某个类层次结构中本来某个虚方法就存在多个覆写版本的话,那么原本也不可能对这个虚方法加上final修饰,因此就算这种状况下去虚化变得困难,锅也不能让“由于没用final修饰”来背。优化
使用final关键字在现代主流的优化JVM上不会提高性能。this
下面是 final 方法的例子:
class User{ public final String getName(){ return "user:wupx"; } } class Reader extends User{ @Override public final String getName(){ return "reader wupx"; //compilation error: overridden method is final } }
使用 final 来修饰的类叫做 final 类,final类一般功能是完整的,它们不能被继承,Java 中有许多类是 final 的,好比 String, Interger 以及其余包装类。下面是 final 类的实例:
final class User{ } class Reader extends User{ //compilation error: cannot inherit from final class }
对于 final 变量,编译器和处理器都要遵照两个重排序规则:
实际上这两个规则也正是针对 final 变量的写与读。写的重排序规则能够保证,在对象引用对任意线程可见以前,对象的 final 变量已经正确初始化了,而普通变量则不具备这个保障;读的重排序规则能够保证,在读一个对象的 final 变量以前,必定会先读这个对象的引用。若是读取到的引用不为空,根据上面的写规则,说明对象的 final 变量必定以及初始化完毕,从而能够读到正确的变量值。
若是 final 变量的类型是引用型,那么构造函数内,对一个 final 引用的对象的成员域的写入,与随后在构造函数外把这个被构造对象的引用赋值给一个引用变量,这两个操做之间不能重排序。
实际上这也是为了保证 final 变量在对其余线程可见以前,可以正确的初始化完成。
下面为使用 final 关键字的一些好处:
参考
《Java编程思想》