深入理解final关键字

在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,它就不是基类接口的一部分,只不过是具有相同名称的方法而已。

  • 11
    点赞
  • 18
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值