final关键字

根据上下文环境,Java的关键字final的含义存在细微的区别,但通常它指的是”这是无法改变的。”不想改变可能出于两种理由:设计或者效率。由于这两个原因相差很远,所以关键字final有可能被误用。

下面来看可能会用到final关键字的三种情况:数据、方法和类。

final数据

对于基本类型,final是数值恒定不变;而对于对象引用,final使引用恒定不变。一旦引用被初始化指向一个对象,就无法再把它改为指向另一个对象。然而,对象其自身确实可以被修改的,Java并未提供使任何对象恒定不变的途径(但可以自己编写类以取得使对象恒定不变的结果)。
下面示范一下final域的情况:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
class Value {
int i;

public Value(int i) {
this.i = i;
}
}

public class FinalData {

private static Random rand = new Random(47);
private String id;

public FinalData(String id) {
this.id = id;
}

private final int valueOne = 9;
//根据惯例,既是static又是final的域,将用大写表示,并使用下划线分隔各个单词
private static final int VALUE_TWO = 99;
private static final int VALUE_THREE = 39;
private final int i4 = rand.nextInt(20);
static final int INT_5 = rand.nextInt(20);
private Value v1 = new Value(11);
private final Value v2 = new Value(22);
private static final Value VAL_3 = new Value(33);

private final int[] a = {1, 2, 3, 4, 5, 6};

public String toString() {
return id + ":" + "i4=" + i4 + ", INT_5 = " + INT_5;
}

public static void main(String[] args) {
FinalData fd1 = new FinalData("fd1");
fd1.v2.i++;
fd1.v1 = new Value(9);
for (int i = 0; i < fd1.a.length; i++)
fd1.a[i]++;
System.out.println(fd1);
System.out.println("Creating new FinalData");
FinalData fd2 = new FinalData("fd2");
System.out.println(fd1);
System.out.println(fd2);
}
}
/* Output
fd1:i4=15, INT_5 = 18
Creating new FinalData
fd1:i4=15, INT_5 = 18
fd2:i4=13, INT_5 = 18
*/

final方法

使用final方法的原因有两个。第一个原因是把方法锁定,以防任何继承类修改它的含义,确保在继承中使方法行为保持不变,并且不会被覆盖。
过去建议使用final方法的第二个原因是效率。在java早期实现中,如果将一个方法指明为final,就是同意编译器将针对该方法的所有调用都转为内嵌调用。但是在最近的java版本中,虚拟机可以探测到这些情况,并优化去掉这些效率反而降低的额外的内嵌调用,因此不能用final方法来优化了。

final类

当将某个类的整体定义为final时,就表明了我们不打算继承这个类,而且也不允许其他人这样做。换句话说,出于某种考虑,我们对该类的设计永不需要做任何变动,或者出于安全的考虑,不希望它有子类。

由于final类禁止继承,final类中的所有方法都隐式的制定为final的。