温故而知新,可以为师矣!( 对所学知识不断复习,不仅可以加深理解,而且可以由此及彼的获得新的知识,就可以做老师了。)

——《论语》

7.6  什么是final

       API 中的某些类,如 String ,以及 Math 等,就是 final 类的典型例子。虽然在 Java 编程中并不经常使用 final 类和 final 方法,但它们有着与众不同的特点,即 final 类不能被继承,不能被覆盖,以及 final 类在执行速度方面比一般类快。下面对 final 类和 final 方法的概念和编程技术分别加以讨论,最后解释为什么 final 类可以提高执行速度。

7.6.1  不能继承final

       有时在程序需要对继承加以限制。例如某些处理特殊运算和操作的类,为了安全理由,不允许被其他类所继承。 final 类没有子类,即它处于继承链的尾部,或者除了自动继承 Object 之外,它们是独立存在的支持类,例如执行密码管理的类,处理数据库信息的管理类等等。
       使用 final 类的另外一个理由是执行速度。由于它的方法不能够被覆盖,所以其地址引用和装载在编译期间完成,而不是在运行期间由 JVM 进行复杂的装载,因而简单和有效。所以如果没有必要,或者不存在有继承的可能性时,尽量使用 final 类。当然,在 API 类库中不多使用 final 类是因为它们是标准程序,希望在实际软件开发中得以广泛使用。而具体的应用软件开发则不同于标准库程序开发。
       注意 final 数据和 final 类的不同。 final 数据指常量,即其值一旦初始化,就不能改变。而 final 类则指不能被其他类所继承的类。

7.6.2  定义final

在类名前加以关键字 final ,这个类就被定义为 final 类,如:

 

public final class SomeClass {
    ...
}

 

或者,

 

public final class SomeClass extends SuperClass {
    ...
}

 

       当一个类被定义为 final 时,它的所有方法都自动成为 final 方法,但不影响对变量的定义。

7.6.3  不能覆盖final方法

也可以在超类中定义某个方法为 final 方法。虽然这个类可以被继承,但子类不能够覆盖 final 方法。 API 类中的许多方法,如 print() println() ,以及 Math 类中的所有方法都定义为 final 方法。在具体应用软件开发中,一些执行特殊性运算和操作的方法,可以定义为 final 方法。在方法的返回类型前加入关键字 final ,则定义该方法为 final ,如:

 

public final String printVersion() {        // 定义 final 方法
    return version;
}

7.6.4  final参数

       final 参数的含义如同 final 变量一样,是常数参数,即当方法接受了这个参数后,其值不能改变。如下代码中定义方法的参数为 final

 

public void setVerison(final String version) {  // 定义常量参数
    this.version = version;
}

 

       在这个方法中使用如下语句产生语法错误:

 

version = "other version…";                     // 非法操作

7.6.5  提高执行速度

       final 类可以提高执行速度主要因为如下原因:
l  不涉及继承和覆盖。
l  其地址引用和装载在编译时完成。
l  在运行时不要求 JVM 执行因覆盖而产生的动态地址引用而花费时间和空间。
l  与继承链上的一般对象相比,垃圾回收器在收回 final 对象所占据的地址空间时也相对简单快捷。
       但在某些情况下使用 final 方法并不能取得提高执行速度的结果。因为并不是所有 final 方法其地址的装载和引用在编译时间完成。
假设类 C 继承了 B B 继承了 A ,在类 A 中有 final 方法。对类 C 来讲,调用 A final 方法的确是 inline 编译,即装载在编译时间完成;但对 A B 来讲,可能没有调用 final 方法。而在执行期间, JVM 动态装载的方法有可能并不是 C 所调用的 final 方法。这种情况下,则不能够取得提高执行速度的结果。当然,如果 final 方法在编译时间装载到 JVM ,而且没有在执行期间覆盖的,可以取得 inline 效益,提高执行速度。
作者建议是:不能仅仅因为考虑追求提高执行速度而使用 final 类。在程序设计和代码编写时,应首先考虑这个类所执行的任务和安全因素,是否允许有子类。在这个前提下,尽量提高代码的重复应用性是面向对象设计和编程的宗旨。然后考虑是否使用 final 类和 final 方法。