阅读原文:小而有料的final(想听文章核心内容语音对话版,请到微信公众号——码上实战,阅读原文)
final这块肉虽然小,但不可不知,因为不仅面试可能会问,实际中也常使用。
就问你常量声明时你为什么要加final?它有哪些作用?你思考过吗?
final通常指被修饰的东西不能被改变的,不能被改变的原因旺往往出于设计和效率的考虑。
先看final能修饰什么?
- 修饰变量
- 修饰方法
- 修饰类
- 修饰参数
不同的修饰有什么不同的作用呢?下面慢慢道来:
final常量
final修饰的常量必须在被使用前进行初始化,初始化的方式无非两种:定义时赋值或构造函数内赋值
final修饰常量基本类型和引用类型有些不同,我分开讲。
基本数据类型
public class Test {
private final int A;
private final int B = 3;
private final int C = new Random().nextInt();
public Test() {
this.A = 5;
}
public Test(int a) {
this.A = a;
}
}
-
为什么常量B要用final修饰?
因为这种确定值(定义便立即赋值)的final常量,编译器会在编译时将该常量值带入到任何可能用到它的计算式中,这会减轻运行时的一些负担。
比如 int d = 3 * B; 那么编译后相当于 int d = 9; -
final修饰的常量值是在编译期间被确定的吗?
注意上面的变量C,我们并不能以为final修饰的就能在编译时知道它的值,C的值只有在运行时被初始化时才会显现。
引用类型
引用类型不能改变指的是引用被初始化指向一个对象后,就再也无法改为指向其他对象,但其指向的对象本身是可以被修改的。
public class B {
//final常量d指向了对象D
private final D d =new D();
//这个方法改变了对象D的内容
public D change(){
this.d.setS("change");
return d;
}
//注意这个方法,将常量d的引用改变了
public D refnewD(){
this.d = new D();
return d;
}
class D {
private String s= "init";
public String getS() {
return s;
}
public void setS(String s) {
this.s = s;
}
}
}
上面代码是无法编译的,因为this.d=new D()会提示Cannot assign a value to final variable 'd'
,但是你却可以调用change()方法来改变d对象的s变量值。
final方法
- 为什么使用final修饰方法
- 设计:把方法锁定,防止继承类修改它的含义。
- 效率:早期编译器会将final方法所有的调用转为内嵌调用,随着JVM的提高,这种方式已被丢弃。
- final和private
当一个方法被private修饰,这会隐式的指定为final,这也会使得子类无法覆盖此方法,可以对private方法增加final修饰,但对该方法并不能增加任何意义。
final类
String类是final应用的典型例子,可去查看面试别再问我String了。
- 为什么要把类指定为final?
出于安全考虑或其他原因,你不希望该类有任何变动或被继承。
一旦类被final修饰,即代表final类中的所有成员变量和方法都会隐式的final.
final参数
在方法的参数列表中将参数指定为final,即代表该参数引用所指向的对象是无法改变的;
若为基本类型,则表示值可以读取,但无法修改。修饰情况和final常量类似。也可结合这题不会!别说你懂值传递与引用传递看。
final相关
static 和 final
- static强调的是该数据只存在一份,且是属于类的,不是属于对象。
- final强调该数据不可变,且是属于对象的。
public class C {
private final double A = 3.14D;
private static double b = 3.14D;
public static void main(String[] args) {
new C();
new C();
new C();
}
}
上面代码运行后,A会存在3个,但b只有一个。
final,finally和finalize
-
finally
是在异常处理时配合
try-catch
执行清理操作,需要清理的资源包括:打开的文件或网络连接等,它会把内存之外的资源恢复到他们的初始状态。无论try中是否有异常出现,finally里的操作都会被执行。 -
finalize
这是Object基类的一个方法,垃圾收集器在将对象清除出内存之前调用的清理资源方法,且此方法只会被系统调用一次,其实finalize能做的工作,try-finally能做的更好,《深入理解Java虚拟机》中建议大家忘掉这个方法的存在。
参考书籍
- Java编程思想
- 深入理解Java虚拟机
更多精彩技术文章尽在微信公众号:码上实战