final表示的是“这部分是无法被改变的”或者是“终态的”。不想被改变的理由有两种,一种是效率,一种是设计。使用到final的一般有三种情况:数据、方法、类。

 

一、final数据:

        有时候数据的恒定不变是很有用的,它能够减轻系统运行时的负担。对于这些恒定不变的数据我可以叫做“常量”。“常量”主要应用与以下两个地方:

       1、编译期常量,永远不可改变。

       2、运行期初始化时,我们希望它不会被改变。

       对于编译期常量,它在类加载的过程就已经完成了初始化,所以当类加载完成后是不可更改的,编译期可以将它代入到任何用到它的计算式中,也就是说可以在编译期执行计算式。当然对于编译期常量,只能使用基本类型,而且必须要在定义时进行初始化。

       有些变量,我们希望它可以根据对象的不同而表现不同,但同时又不希望它被改变,这个时候我们就可以使用运行期常量。对于运行期常量,它既可是基本数据类型,也可是引用数据类型。基本数据类型不可变的是其内容,而对于对象引用,不能改变的是他的引用,而对象本身是可以修改的。因为引用类型变量里面放的是个地址,所以用final修饰引用类型变量指的是它里面的地址不能变,并不是说这个地址所指向的对象或数组的内容不可以变:

      例如:类中有一个属性是final Person p=new Person("name"); 那么你不能对p进行重新赋值,但是可以改变p里面属性的值,p.setName('newName');

       final修饰属性,声明成员变量时可以不赋值,而且一旦赋值就不能被修改了。对final属性可以在三个地方赋值:声明时、初始化块中、构造方法中。总之一定要赋值。 (如果修饰的是类变量:必须要在静态初始化块中指定初始值或者声明该类变量时指定初始值,而且只能在这两个地方之一进行指定;如果声明的是局部变量:如果final局部变量未进行初始化,可以进行赋值,当且仅有一次赋值,一旦赋值之后再次赋值就会出错)

 

二、final方法:

当父类的方法被final修饰时,子类不能重写父类的该方法,但可以被子类继承或者重载。所以finali方法使用的第一个原因就是方法锁定。比如在Object中,getClass()方法就是final的,我们就不能重写该方法,但是hashCode()方法就不是被final所修饰的,我们就可以重写hashCode()方法。

第二个原因就是效率问题:在《Java编程思想》中,在java的早期实现中,如果将一个方法指明为final,就是同意编译器将针对该方法的所有调用都转为内嵌调用。当编译器发现一个final方法的调用命令时,它会根据自己的谨慎判断,跳过插入程序代码这种正常的调用方式而执行方法调用机制(将参数压入栈,跳至方法代码处执行,然后跳回并清理栈中的参数,处理返回值),并且以方法体中的实际代码的副本来代替方法调用。这将消除方法调用的开销。当然,如果一个方法很大,你的程序代码会膨胀,因而可能看不到内嵌所带来的性能上的提高,因为所带来的性能会花费于方法内的时间量而被缩减。

通俗一点说就是:当一个方法被修饰为final方法时,意味着编译器可能将该方法用内联(inline)方式载入,所谓内联方式,是指编译器不用像平常调用函数那样的方式来调用方法,而是直接将方法内的代码通过一定的修改后copy到原代码中。这样可以让代码执行的更快(因为省略了调用函数的开销)。inline需要在编译的时候就知道最后要用哪个函数, 显然,非final是不行的。非final方法可能在子类中被重写,由于可能出现多态的情况,编译器在编译阶段,并不能确定将来调用方法的对象的真正类型,也就无法确定到底调用哪个方法。

父类的final方法是不能被子类所覆盖的,也就是说子类是不能够存在和父类一模一样的方法的:

public class Custom extends Person{
    public void method1(){
        System.out.println("Person's  method1....");
    }

//    Cannot override the final method from person:子类不能覆盖父类的final方法
//    public void method2(){
//        System.out.println("Person's method2...");
//    }
}

注意:final不能用于修饰构造方法。 
父类的private成员方法是不能被子类方法覆盖的,因此private类型的方法默认是final类型的。

 

三、final类:

final类不能被继承,没有子类,final类中所有方法都是final的。子类继承往往可以重写父类的方法和改变父类属性,会带来一定的安全隐患,因此,当一个类不希望被继承时就可以使用final修饰。对于final修饰的类来说,它的成员变量可以为final,也可以为非final。如果定义为final,那么final数据的规则同样适合它,而它的方法则会自动的加上final,因为final类是无法被继承,所以这个是默认的。

 

四、final参数:

 在实际应用中,我们除了可以用final修饰成员变量、成员方法、类,还可以修饰参数、若某个参数被final修饰了,则代表了该参数是不可改变的。final修饰参数在内部类中是非常有用的,在匿名内部类中,为了保持参数的一致性,若所在的方法的形参需要被内部类里面使用时,该形参必须为final

如果在方法中我们修改了该参数,则编译器会提示你:The final local variable i cannot be assigned. It must be blank and not using a compound assignment。

public class Custom {
    public void test(final int i){
      //i++;     ---final参数不可改变
        System.out.println(i);
    }

    public void test(final Person p){
     //p = new Person();    --final参数不可变
     p.setName("chenssy");
    }
}

 

小结:
 
final成员变量表示常量,只能被赋值一次,赋值后值不能再改变。 

final方法不能被子类的方法覆盖,但可以被继承。

final类不能被继承,没有子类,final类中的方法默认是final的。 

 

参考博客:https://blog.csdn.net/chenssy/article/details/17591563

10-07 19:08