概念

首先final翻译过来就是:最终的;
它是一个修饰符,用在不同的地方作用不同,但是本质又很相似。

它可以用来修饰一下几种:

    • 不可被继承
  • 方法
    • 不可被子类覆盖
  • 变量
    • 不可被重新赋值「这个最难」

具体说明:

当final修饰的时候代表这个内容是不可以被继承的,就是当前是什么样子就最终是什么样子了。

为什么学final

前面学了继承,但是对应的继承还是有弊端的。它打破封装性,就是我们本来封装好的方法或者类,如果被继承了,就可以重写对应的类里面方法的实现逻辑了。这样就会有一个问题,如果恶意继承某个类并进行不正确的覆盖,那会导致原来的功能错误

比如说,下面这个是一个父类,对应实现了一个sleep()方法

  1. public class Animal {
  2. public void sleep(){
  3. System.out.println("父类Animal的sleep方法...");
  4. }
  5. }

下面是一个子类,继承了上面的Animal类后重写了对应的sleep()方法,那么在创建实例并调用sleep()的时候,真正方法的实现就是子类Dog里面的sleep()方法实现,而非父类Animal的sleep()方法的实现内容。

  1. //子类
  2. public class Dog extends Animal {
  3. public void sleep(){
  4. System.out.println("子类Dog的sleep方法...");
  5. }
  6. }

进行对应的测试验证:

  1. public class FinalTest {
  2. public static void main(String[] args) {
  3. Dog dog = new Dog();
  4. dog.sleep();
  5. }
  6. }

测试代码实际最后运行的是子类Dog的sleep()方法。
关键字:final - 图1

final修饰类

修饰的类不可以被继承。

final用来修饰一个类的时候,代表这个类不能被其他的类继承 extends;不能再对这个类进行改变

  • 比如java源码里面的 String类 System类 StringBuffer类
  • java不允许别人去扩展这个String类或者重写这个类,如果想要重写这个类是不可以的
  • final 修饰的类不能有子类,但是可以有父类

关键字:final - 图2

回到上面的例子,如果想要父类的实现内容不被重写,则直接在父类声明上添加对应的final关键字,代码如下:

  1. //public class Animal {
  2. //被final修饰的类不能被继承 Animal这个类就不能被其他的类extends
  3. public final class Animal {
  4. public void sleep(){
  5. System.out.println("父类Animal的sleep方法...");
  6. }
  7. }

关键字:final - 图3
对应Dog类就无法进行继承,直接就是新的一个类进行重新编写。

  1. //public class Animal extends Animal {
  2. //被final修饰的类是不可以被继承的 Animal不能被Dog继承
  3. public class Dog {
  4. //重新一个类的方法
  5. public void sleep(){
  6. System.out.println("Dog的sleep方法...");
  7. }
  8. }

总结

  • final修饰的类是不能被继承的
  • final修饰的类可以继承其他类

修饰方法

修饰的方法不可被覆盖。

final用来修饰方法的时候,表示这个方法不能被重写,代表这是这个方法最终的一个形式

  • 比如java里面的Object类「是所有类的父类」,它其中的getClass()方法就是final修饰的

关键字:final - 图4

修饰构造方法

  • final修饰构造方法会直接报错
    • 因为final不能修饰构造方法

别的方法可以正常覆盖,但是构造方法不同,构造方法必须和类名一致,就无法直接覆盖;「子类也没有诉求说去覆盖父类的构造方法」
所以final放在构造方法上也没有意义。

关键字:final - 图5

  1. //被final修饰的类不能被继承 Animal这个类就不能被其他的类extends
  2. //public class Animal {
  3. public final class Animal {
  4. public void sleep(){
  5. System.out.println("父类Animal的sleep方法...");
  6. }
  7. //final修饰构造方法直接报错
  8. public final Animal() {
  9. }
  10. }

修饰一般方法

想要继承父类但是对应方法又不想被重写,则在不想重写的方法使用final关键字进行修饰。

  1. public class Animal {
  2. //对应方法不想重新实现,直接在方法上添加对应final
  3. public final void eat(){
  4. System.out.println("父类Animal的eat方法...");
  5. }
  6. public void sleep(){
  7. System.out.println("父类Animal的sleep方法...");
  8. }
  9. //final修饰构造方法直接报错
  10. public /*final*/ Animal() {
  11. }
  12. }

对应子类的实现方法则会报错,报错信息如下图:
关键字:final - 图6

所以子类对应的eat()方法进行注释掉,然后重写sleep()是没问题的

  1. public class Dog extends Animal {
  2. //因为Animal类中的eat方法被final修饰,所以在子类Dog 中无法进行重写
  3. /*
  4. public void eat(){
  5. //父类的eat方法被覆盖了
  6. System.out.println("子类Dog的eat方法...");
  7. }
  8. */
  9. // 没有使用final修饰的方法可以重写
  10. public void sleep(){
  11. System.out.println("子类Dog的sleep方法...");
  12. }
  13. }

总结

  • 子类无法重新加载final修饰的方法
  • 不要随便定义final方法,可能会带来很多不必要的问题

如果父类有些行为就很确定,不会进行更改,那该行为的方法可以使用final修饰

修饰变量

修饰的变量是一个常量,只能被赋值一次。
final 修饰变量,这个时候变量就叫常量了。

如果再次赋值,则对应代码报错,提示我们该变量不能被final修饰

关键字:final - 图7

正常声明的变量可以被再次赋值,只有final修饰的变量不能被再次赋值

  1. int age = 26;
  2. age = 33;
  3. System.out.println("年龄是:"+age);
  4. //final修饰的变量
  5. final int number = 6;
  6. // number = 9;//java: 无法为最终变量number分配值
  7. System.out.println("对应礼品个数是:"+number);

赋值位置

final修饰变量的时候对应的赋值位置「就是初始化变量」有以下3种:

显示初始化

代码块中初始化

构造器中初始化

一般成员变量

首先第一个我们先来看一下对应的final修饰一般成员变量,这个就有2种赋值方式:

直接赋值「显示初始化」

第一种是直接赋值,就是在声明的时候,直接对声明的成员变量进行赋值操作,也叫做显示初始化。

  1. package top.testeru.keywords.finalp;
  2. //被final修饰的类不能被继承 Animal这个类就不能被其他的类extends
  3. public class Animal {
  4. //public final class Animal {
  5. //对应名称
  6. private String name;
  7. private int sex = 1;//0代表公,1代表母
  8. //多大
  9. private int age;
  10. //对应方法不想重新实现,直接在方法上添加对应final
  11. public final void eat(){
  12. System.out.println("父类Animal的eat方法...");
  13. }
  14. public void sleep(){
  15. System.out.println("父类Animal的sleep方法...");
  16. }
  17. //final修饰构造方法直接报错
  18. public /*final*/ Animal() {
  19. }
  20. public Animal(String name, int sex, int age) {
  21. this.name = name;
  22. this.sex = sex;
  23. this.age = age;
  24. }
  25. }
  • 对应sex变量用final修饰后,表示该变量不可被重新赋值,所以构造方法中对应sex变量赋值会报错,解决方案就是把红框内代码删除就好了

构造方法删掉该参数传入

关键字:final - 图8

  1. public class Animal {
  2. //对应名称
  3. private String name;
  4. private final int sex = 1;//0代表公,1代表母
  5. //多大
  6. private int age;
  7. //对应方法不想重新实现,直接在方法上添加对应final
  8. public final void eat(){
  9. System.out.println("父类Animal的eat方法...");
  10. }
  11. public void sleep(){
  12. System.out.println("父类Animal的sleep方法...");
  13. }
  14. //final修饰构造方法直接报错
  15. public /*final*/ Animal() {
  16. }
  17. /* public Animal(String name, int sex, int age) {
  18. this.name = name;
  19. this.sex = sex;
  20. this.age = age;
  21. }*/
  22. public Animal(String name, int age) {
  23. this.name = name;
  24. this.age = age;
  25. }
  26. }

构造方法赋值「构造器中初始化」

  • 也可以直接声明成员变量,然后在对应的构造方法内赋值,也是只有一次,在每一次进行构造声明的时候进行赋值。
  • 可以声明多个对象,进行多个赋值,只不过对应对象不同,对应内存地址也不同
  1. public class Animal {
  2. //对应名称
  3. private String name;
  4. private final int sex = 1;//0代表公,1代表母
  5. //多大
  6. private final int age;
  7. //对应方法不想重新实现,直接在方法上添加对应final
  8. public final void eat(){
  9. System.out.println("父类Animal的eat方法...");
  10. }
  11. public void sleep(){
  12. System.out.println("父类Animal的sleep方法...");
  13. }
  14. public Animal(String name, int age) {
  15. this.name = name;
  16. //构造函数初始化的时候进行一次赋值,后期无法再次进行更改
  17. this.age = age;
  18. }
  19. }

关键字:final - 图9

验证的代码

  1. Animal animal1 = new Animal("小黑",2);
  2. System.out.println(animal1);
  3. Animal animal2 = new Animal("小白",3);
  4. System.out.println(animal2);

运行的结果:

  1. Animal{name='小黑', sex=1, age=2}
  2. Animal{name='小白', sex=1, age=3}

关键字:final - 图10

总结:

以上2种声明只能用其中一种,而且必须用2种的其中一个给它进行赋值,不然也是会报错。「不赋值会报错」

final 就是过了这个村就没这个店了,所以你一定要在构造方法里赋值,并且只赋值一次。

静态成员变量

final修饰静态成员变量,这个有2种赋值方式:

直接赋值「显示初始化」

第一种是直接赋值,就是在声明的时候,直接对声明的静态成员变量进行赋值操作,也就是显示初始化。

  1. //和成员变量一样,一定要赋值并且只赋值一次
  2. //动物最长的寿命 60年
  3. static final int MAX_AGE = 60;

在static代码块赋值「代码块中初始化」

  1. //动物最短寿命1 年
  2. static final int MIN_AGE;
  3. //只能赋值一次,而且必须要赋值一次
  4. static{
  5. MIN_AGE = 1;
  6. }
  • 在其它的方法里面赋值是不可以的

调用的时候是直接使用类名.变量名进行静态变量的调用

关键字:final - 图11

总结

这个时候只要赋值的地方不是静态代码块内,无论你在哪个方法内添加对应的赋值,在声明变量的那一行都会报错,因为它会认为你都没有给我去赋值

final修饰形参

final修饰对应形参的时候,说明这个形参是一个常量。当我们调用这个方法的时候,给常量形参赋上一个实际的参数值。

注意⚠️:形参被修饰后,在方法内实现业务逻辑代码中,无法再对该形参进行重新赋值,只能调用该参数。

形参也是只能赋值一次,在调用类的.方法的时候传入对应值。

  1. public void sleep(final int time){
  2. System.out.println("大概几点开始睡觉:"+ time);
  3. // time = 0;
  4. System.out.println("父类Animal的sleep方法...");
  5. }

关键字:final - 图12

形参一旦赋值了以后,就只能在方法体内使用这个形参,但是不能进行重新的赋值。

final修饰局部变量

直接赋值「显示初始化」

直接赋值就是在声明的时候,直接对声明的局部变量进行赋值操作,也就是显示初始化。

  1. public void sleep(final int time){
  2. System.out.println("大概几点开始睡觉:"+ time);
  3. // time = 0;
  4. //一般局部变量
  5. String selfHouse = "米小圈的家";
  6. selfHouse = house();
  7. final String otherHouse = "动物之家";
  8. //赋值后无法进行再次赋值的业务逻辑
  9. otherHouse = house();
  10. System.out.println("父类Animal的sleep方法...");
  11. }
  12. private String house(){
  13. return "My House";
  14. }

关键字:final - 图13

调用时赋值

  1. public void sleep(final int time){
  2. System.out.println("大概几点开始睡觉:"+ time);
  3. // time = 0;
  4. //一般局部变量
  5. String selfHouse = "米小圈的家";
  6. selfHouse = house();
  7. final String otherHouse = "动物之家";
  8. //赋值后无法进行再次赋值的业务逻辑
  9. otherHouse = house();
  10. //final修饰声明的局部变量直接声明,在后面调用的时候赋值
  11. final String otherHouse1;
  12. otherHouse1 = house();
  13. System.out.println("父类Animal的sleep方法...");
  14. }
  15. private String house(){
  16. return "My House";
  17. }

什么时候修饰变量?

在程序中一些不会发生变化的数据,就是常量,比如:3.14。
直接写的时候代码复用性不高,也不利于阅读,所以一般就会给这个常量数据起一个相对容易阅读的变量名。

  1. final double PI = 3.14;

关键字:final - 图14

  • static final
    • 修饰属性:全局常量
    • 修饰方法「很少用,可用」
    • 这2个关键字不冲突
  • abstract final
    • 关键字冲突

final 有一个引用 纸上的内容可以擦了再写擦了再写,但是纸对应的地址,是第几张纸,这个是不可以进行更改的

应用

单例模式-饿汉式

  1. package top.testeru.keywords;
  2. /**
  3. * @Package: top.testeru.keywords
  4. * @author: testeru.top
  5. * @Description: 单例模式,饿汉式【工作中常用】
  6. * 类加载的时候就创建对象
  7. * @date: 2022年01月26日 1:58 PM
  8. */
  9. public class Single {
  10. //1.私有化构造函数
  11. private Single() {}
  12. //2.创建一个私有并且静态的本类对象
  13. private static final Single s = new Single();
  14. //3.创建一个公共的static方法返回该对象
  15. public static Single getInstance(){
  16. return s;
  17. }
  18. void show(){
  19. System.out.println("这是单例的一个方法");
  20. }
  21. }
  1. package top.testeru.keywords;
  2. /**
  3. * @Package: top.testeru.keywords
  4. * @author: testeru.top
  5. * @Description:
  6. * @date: 2022年01月26日 2:06 PM
  7. */
  8. public class SingleTest {
  9. public static void main(String[] args) {
  10. Single single = Single.getInstance();
  11. single.show();
  12. }
  13. }

final面试题

题目一

  1. public class FinalInterView {
  2. public int add(final int x){
  3. //return ++x;//编译报错,相当于给x重新赋值
  4. return x+1;
  5. }
  6. }
  • ++x 相当于给x进行了重新赋值,对应的编译就会报错
  • x+1 x对应的值不更改,只是返回x+1后的结果
  • 只要x的值不变

题目二

  1. public class FinalInterView {
  2. public static void main(String[] args) {
  3. Other o = new Other();
  4. new FinalInterView().addOne(o);
  5. }
  6. private void addOne(final Other o) {
  7. // o = new Other();//o传入过来的就是一个不变的对象,不能重新new
  8. System.out.println(o.i+6);//虽然o这个对象不能变了但是这个对象的一些属性和方法是可以变的
  9. //比如说,我选中了你是我一辈子的朋友,但是你的一些工资收入以及年龄大小这些属性和工作内容的方法是可以进行变化的
  10. }
  11. }
  12. class Other{
  13. public int i;
  14. }

关键字:final - 图15