深入理解Java-final关键字
1 修饰属性或者变量
无论属性是基本类型还是引用类型,作用都是变量里面存放的“值”不可变
经常和static关键字一起使用,作为常量
- 基本类型,变量放的是实实在在的值,如1,“abc”等
- 引用类型,变量放的是个地址,所以用final修饰引用类型变量指的是它里面的地址不能变,即它只能指向初始时指向的那个对象,而不关心指向的对象内容的变化
所以修饰的变量必须被初始化
public static final String LOAN = "loan";
LOAN = new String("loan") //invalid compilation error
- 定义时
- 初始化块中,但不可在静态初始化块中,静态的final实例变量才可以在静态初始化块中
- 构造方法中,但静态final实例变量不可以在其中
final变量是只读的
2 修饰方法
该方法可被继承,但不许被任何子类重写
当调用final方法时,直接将方法主体插入到调用处,而不是进行方法调用,这样能提高程序效率(内联机制)
如果你认为一个方法的功能已经足够完整了,子类中不需要改变的话,你可以声明此方法为final
final方法比非final方法快,因为在编译时候已静态绑定了,不需要在运行时再动态绑定。
class PersonalLoan{
public final String getName(){
return "personal loan";
}
}
class CheapPersonalLoan extends PersonalLoan{
@Override
public final String getName(){
return "cheap personal loan"; //compilation error: overridden method is final
}
}
3 修饰类
使用final来修饰的类叫作final类
final类通常功能是完整的,不能被继承
Java中有许多类是final的,譬如String, Interger以及其他包装类
类不可以被继承,但这并非表示final类的实例变量也不可变,除非给实例变量也增加final修饰
final class PersonalLoan{
}
class CheapPersonalLoan extends PersonalLoan{ //compilation error: cannot inherit from final class
}
一个类不可同时被abstract和final修饰
思考一个有趣的现象:
byte b1=1;
byte b2=3;
byte b3=b1+b2; //当程序执行到这一行的时候会出错,因为b1、b2可以自动转换成int类型的变量,运算时java虚拟机对它进行了转换,结果导致把一个int赋值给byte
final byte b1=1;
final byte b2=3;
byte b3=b1+b2; //不会出错,相信你看了上面的解释就知道原因了。
4 final关键字的好处
- 提高性能
JVM和Java应用都会缓存final变量。 - final变量可以安全的在多线程环境下进行共享,而不需要额外的同步开销。
- 使用final关键字,JVM会对方法、变量及类进行优化。
5 不可变类
创建不可变类要使用final关键字。不可变类是指它的对象一旦被创建了就不能被更改了。String是不可变类的代表。不可变类有很多好处,譬如它们的对象是只读的,可以在多线程环境下安全的共享,不用额外的同步开销等等。
6其他重要知识点
不能够对final变量再赋值
本地变量必须在声明时赋值
在匿名类中所有变量都必须final
final关键字不同于finally关键字,后者用于异常处理
final关键字容易与finalize()方法搞混,后者是在Object类中定义的方法,是在垃圾回收之前被JVM调用的方法
接口中声明的所有变量本身是final
final和abstract这两个关键字反相关,final类不能abstract
final方法在编译阶段绑定,称为静态绑定(static binding)
没在声明时初始化final变量的称为空白final变量(blank final variable),须在构造器中初始化,或调用this()初始化。不这么做的话,编译器会报错“final变量(变量名)需要进行初始化”
final变量就是常量,而且通常常量名要大写
private final int COUNT = 10;
对于集合对象声明为final指的是引用不能被更改,但是你可以向其中增加,删除或者改变内容。譬如:
private final List Loans = new ArrayList();
list.add(“home loan”); //valid
list.add("personal loan"); //valid
loans = new Vector(); //not valid