我去,你居然还不会用 Java 的 final 关键字

写一篇文章容易吗?太不容易了,首先,须要一个安静的环境,这一点就很是不容易。不少小伙伴的办公室都是开放式的,很是吵,何况上班时间写的话,领导就不高兴了;只能抽时间写。其次,环境有了,还要有一颗安静的心,若是内心装着其余挥之不去的事,那就糟糕了,呆坐着电脑前一成天也不会有结果。html

我十分佩服一些同行,他们写万字长文,这在我看来,几乎不太可能完成。由于我要日更,一万字的长文,若是走原创的话,至少须要一周时间,甚至一个月的时间。java

就如小伙伴们看到的,我写的文章大体都能在五分钟内阅读完,而且可以保证小伙伴们在阅读完学到或者温习到一些知识。这就是个人风格,通俗易懂,轻松幽默。git

好了,又一篇我去系列的文章它来了:你居然还不会用 final 关键字。程序员

已经晚上 9 点半了,我尚未下班,由于要和小王一块修复一个 bug。我订了一份至尊披萨,和小王吃得津津有味的时候,他忽然问了我一个问题:“老大,能给我详细地说说 final 关键字吗,总感受对这个关键字的认知不够全面。”github

一会儿个人火气就来了,尽管小王问的态度很谦逊,很卑微,但我仍是忍不住破口大骂:“我擦,小王,你丫的居然不会用 final,我当初是怎么面试你进来的!”面试

发火归发火,我这我的仍是有原则的,等十点半回到家后,我决定为小王专门写一篇文章,好好地讲一讲 final 关键字,也但愿给更多的小伙伴一些帮助。编程

尽管继承可让咱们重用现有代码,但有时处于某些缘由,咱们确实须要对可扩展性进行限制,final 关键字能够帮助咱们作到这一点。安全

0一、final 类

若是一个类使用了 final 关键字修饰,那么它就没法被继承。若是小伙伴们细心观察的话,Java 就有很多 final 类,好比说最多见的 String 类。微信

public final class String implements java.io.Serializable, Comparable<String>, CharSequence, Constable, ConstantDesc {}
复制代码

为何 String 类要设计成 final 的呢?缘由大体有如下三个:并发

  • 为了实现字符串常量池
  • 为了线程安全
  • 为了 HashCode 的不可变性

更详细的缘由,能够查看我以前写的一篇文章

任未尝试从 final 类继承的行为将会引起编译错误,为了验证这一点,咱们来看下面这个例子,Writer 类是 final 的。

public final class Writer {
    private String name;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
}
复制代码

尝试去继承它,编译器会提示如下错误,Writer 类是 final 的,没法继承。

不过,类是 final 的,并不意味着该类的对象是不可变的。

Writer writer = new Writer();
writer.setName("沉默王二");
System.out.println(writer.getName()); // 沉默王二
复制代码

Writer 的 name 字段的默认值是 null,但能够经过 settter 方法将其更改成“沉默王二”。也就是说,若是一个类只是 final 的,那么它并非不可变的所有条件。

若是,你想了解不可变类的所有真相,请查看我以前写的文章此次要说不明白immutable类,我就怎么地。忽然发现,写系列文章真的妙啊,不少相关性的概念所有涉及到了。我真服了本身了。

把一个类设计成 final 的,有其安全方面的考虑,但不该该故意为之,由于把一个类定义成 final 的,意味着它没办法继承,假如这个类的一些方法存在一些问题的话,咱们就没法经过重写的方式去修复它。

0二、final 方法

被 final 修饰的方法不能被重写。若是咱们在设计一个类的时候,认为某些方法不该该被重写,就应该把它设计成 final 的。

Thread 类就是一个例子,它自己不是 final 的,这意味着咱们能够扩展它,但它的 isAlive() 方法是 final 的:

public class Thread implements Runnable {
    public final native boolean isAlive();
}
复制代码

须要注意的是,该方法是一个本地(native)方法,用于确认线程是否处于活跃状态。而本地方法是由操做系统决定的,所以重写该方法并不容易实现。

Actor 类有一个 final 方法 show()

public class Actor {
    public final void show() {
        
    }
}
复制代码

当咱们想要重写该方法的话,就会出现编译错误:

若是一个类中的某些方法要被其余方法调用,则应考虑事被调用的方法称为 final 方法,不然,重写该方法会影响到调用方法的使用。

一个类是 final 的,和一个类不是 final,但它全部的方法都是 final 的,考虑一下,它们之间有什么区别?

我能想到的一点,就是前者不能被继承,也就是说方法没法被重写;后者呢,能够被继承,而后追加一些非 final 的方法。没毛病吧?看把我聪明的。

0三、final 变量

被 final 修饰的变量没法从新赋值。换句话说,final 变量一旦初始化,就没法更改。以前被一个小伙伴问过,什么是 effective final,什么是 final,这一点,我在以前的文章也有阐述过,因此这里再贴一下地址:

www.itwanger.com/java/2020/0…

1)final 修饰的基本数据类型

来声明一个 final 修饰的 int 类型的变量:

final int age = 18;
复制代码

尝试将它修改成 30,结果编译器生气了:

2)final 修饰的引用类型

如今有一个普通的类 Pig,它有一个字段 name:

public class Pig {
   private String name;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
}
复制代码

在测试类中声明一个 final 修饰的 Pig 对象:

final Pig pig = new Pig();
复制代码

若是尝试将 pig 从新赋值的话,编译器一样会生气:

但咱们仍然能够去修改 Pig 的字段值:

final Pig pig = new Pig();
pig.setName("特立独行");
System.out.println(pig.getName()); // 特立独行
复制代码

3)final 修饰的字段

final 修饰的字段能够分为两种,一种是 static 的,另一种是没有 static 的,就像下面这样:

public class Pig {
   private final int age = 1;
   public static final double PRICE = 36.5;
}
复制代码

非 static 的 final 字段必须有一个默认值,不然编译器将会提醒没有初始化:

static 的 final 字段也叫常量,它的名字应该为大写,能够在声明的时候初始化,也能够经过 static 代码块初始化

  1. final 修饰的参数

final 关键字还能够修饰参数,它意味着参数在方法体内不能被再修改:

public class ArgFinalTest {
    public void arg(final int age) {
    }

    public void arg1(final String name) {
    }
}
复制代码

若是尝试去修改它的话,编译器会提示如下错误:

0四、总结

亲爱的读者朋友,我应该说得很全面了吧?我想小王看到了这篇文章后必定会感谢个人良苦用心的,他毕竟是个积极好学的好同事啊。

若是以为文章对你有点帮助,请微信搜索「 沉默王二 」第一时间阅读,回复「并发」更有一份阿里大牛重写的 Java 并发编程实战,今后不再用担忧面试官在这方面的刁难了。

本文已收录 GitHub,传送门~ ,里面更有大厂面试完整考点,欢迎 Star。

我是沉默王二,一枚有颜值却靠才华苟且的程序员。关注便可提高学习效率,别忘了三连啊,点赞、收藏、留言,我不挑,嘻嘻

相关文章
相关标签/搜索