| 一 概念 在Java中, 可以使用final关键字修饰类、方法以及成员变量。
 
 1 final标记的类不能被继承
 在设计类时候,如果这个类不需要有子类,类的实现细节不允许改变,并且确信这个类不会载被扩展,那么就设计为final类。
 
 2 final标记的方法不能被子类重写
 ① 把方法锁定,防止任何继承类修改它的意义和实现
 ② 高效。编译器在遇到调用final方法时候会转入内嵌机制,大大提高执行效率
 
 3 final标记的变量即成为常量,只能被赋值一次
 
 二 用法
 
 1 对于引用类型和值类型来说
 final的变量一旦被赋值,就不能再此改变。但对于引用类型,其“不可改变”的是指其地址不能改变而其属性可以改变。
 
 
 
 2 对于成员变量和局部变量来说
 对于final的成员变量,对其赋值有两种方式:① 声明时即赋值;②构造方法中赋值。
 对于final的局部变量,对其赋值有两种方式:①声明时即赋值;②使用的时候赋值。
 
 例子一
 public class Test
 {
 final int a;
 final int b = 0;//声明时赋值
 public Test()
 {
 a = 0;
 // b = 1;//编译不能通过
 }
 public Test(int i)
 {
 a = 1;
 }
 public static void printFinal()
 {
 final int m = 100;
 // m = 101;//编译不能通过
 final int n;
 n = 102;
 // n = 103;
 System.out.println("m + n=" + m + n);
 }
 }
 
 例子二
 import cn.xy.model.Person;
 public class Test2
 {
 final Person p = new Person("xy");
 final Person p2;
 public Test2()
 {
 p2 = new Person("xy");
 p.setName("xyy");
 // p = new Person("Jim");//编译不能通过
 // p = null;
 }
 public Test2(int i)
 {
 p2 = new Person("xy");
 }
 public static void printFinal()
 {
 final Person p = new Person("xy");
 p.setName("xyy");
 // p = new Person("Jim");
 // p = null;
 }
 }
 
 3 关于final修饰类和方法的例子这里不举例,读者可以试一试。
 
 三 final内存分配
 
 调用一个函数除了函数本身的执行时间之外,还需要时间去寻找这个函数(类内部有一个函数签名和函数地址的映射表)。故减少函数调用次数就等于降低了性能消耗。final修饰的函数会被编译器优化,优化的结果是减少了函数调用的次数。
 
 public class Test3
 {
 final void function()
 {
 System.out.println("xy");
 }
 
 public static void main(String[] args)
 {
 Test3 t = new Test3();
 for(int i = 0;i< 1000;i++)
 {
 t.function();
 }
 }
 }
 
 经过编译器优化之后,这个类变成了相当于这样写:
 public class Test3
 {
 final void function()
 {
 System.out.println("xy");
 }
 
 public static void main(String[] args)
 {
 Test3 t = new Test3();
 for(int i = 0;i< 1000;i++)
 {
 System.out.println("xy");
 }
 }
 }
 优点:
 编译器直接将function的函数体内嵌到了调用函数的地方,这样的结果是节省了1000次函数调用,当然编译器处理成字节码,只是我们可以想象成这样,看个明白。
 
 缺点:
 当函数体若太长时用final会适得其反,因为经过编译器内嵌之后代码长度大大增加,于是就增加了jvm解释字节码的时间。而且如果final修饰的方法体过大的话,编译器可能会放弃内联。
 
 |