深入理解final关键字

    xiaoxiao2021-03-25  85

    在Java中,final关键字可以用来修饰数据、方法、参数、类,下面我们来了解final的关键字的用法。

    基本用法

    final 数据

    对于基本类型,final使数值恒定不变;而对用对象引用,final使引用恒定不变。    final修饰的基本类型,一旦被初始化后,不能再被赋值。    final修饰的对象引用,一旦引用被初始化指向一个对象,就无法再把它改为指向另外一个对象。

    final 类

    当用final修饰一个类时,表明这个类不能被继承。也就是说,如果一个类你永远不会让他被继承,就可以用final进行修饰。    final类中的成员变量可以根据需要设为final,但是要注意final类中的所有成员方法都会被隐式地指定为final方法。

    final 方法

    “使用final方法的原因有两个。第一个原因是把方法锁定,以防任何继承类修改它的含义;第二个原因是效率。在早期的Java实现版本中,会将final方法转为内嵌调用。但是如果方法过于庞大,可能看不到内嵌调用带来的任何性能提升。在最近的Java版本中,不需要使用final方法进行这些优化了。

    public class Person { *** public final void doSwim() { } *** } ![这里写图片描述](https://img-blog.csdn.net/20161016140636177)

    因此,如果只有在想明确禁止 该方法在子类中被覆盖的情况下才将方法设置为final的。

    注:类的private方法会隐式地被指定为final方法。

    final 参数

    使用final修饰方法的参数,若参数为基本类型,该参数不能在方法中修改其值。若参数为对象应用,该参数在方法中不能修改其指向引用。 

    深入理解final

    空final

    class Reapeat { private final int a = 10; private final int b; public Reapeat(int b) { this.b = b; } }

    所谓的”空白final”是指被声明的为final但又为给定初值的对象引用或者基本数据。无论在什么情况下,编译器都会去确保final在使用前必须被初始化。若不进行初始化,会提示错误,如下:    意味着,必须在域的定义处或者每个构造器中使用表达式对final进行赋值,这正是final域在使用前被初始化的原因所在。

    final和static

    public class FinalData { private static Random random = new Random(47); public final int v_1 = random.nextInt(20); public final static int v_2 = random.nextInt(20); } public class TestFinal { public static void main(String[] args) { FinalData fd_1 = new FinalData(); FinalData fd_2 = new FinalData(); System.out.println("v_1 = " + fd_1.v_1 + " | v_2 = " + FinalData.v_2); System.out.println("v_1 = " + fd_2.v_1 + " | v_2 = " + FinalData.v_2); } } // Log v_1 = 15 | v_2 = 18 v_1 = 13 | v_2 = 18

    从上述代码可以看出,final将数值定义为静态和非静态的区别。在fd_1和fd_2中,v_1的值都是唯一的,而v_2的值是一致的,并没有因为对象的创建而加以改变,因为其被static修饰,意味着在装载时已被初始化,而不是每次创建新对象时都初始化。

    final和对象引用

    我们已经了解到,如果使用final修饰了引用对象,引用对象被初始化后,不能再被指向另外一个对象,但是其内部的内容是否可以修改?

    final Person person = new Person("li", 20); person.setName("wang"); person.setAge(40); System.out.println(person.toString()); // Log打印 Person{name='wang', age=40}

    从上述代码,person对象被final修饰,同时初始化为name = “li”,age=”20”。然后调用其相应setXX()方法修改其值。从Log打印看出,person对象自身的成员值为修改后的值。意味着被final修饰的对象引用,只是对象的应用不能修改,但是其自身却是可以修改的。

    final和private

    类中所有的private方法都隐式的指定为final的,由于无法取用private方法,所以也就无法覆盖它,可以对private方法添加final修饰符,但并没有添加任何额外意义。

    private final void doSwim() { }

    “覆盖”即重写只有在某方法是基类的接口的一部分时才会出现,即必须将一个对象向上转型为它的基本类型并调用相同的方法。如果某方法为private,它就不是基类接口的一部分,只不过是具有相同名称的方法而已。

    转载请注明原文地址: https://ju.6miu.com/read-11381.html

    最新回复(0)