怎么声明一个类不会被继承,以及应用场景
1. 概述
今天的
Java 面试合集
又来了。今天我们复习的问题是:怎么声明一个类,不可以被继承
2. 验证
public final class TestMath {
}
通过上述截图 我们可以看到,被关键字final
修饰过的类,是不能被继承的。从语法层面就无法通过。
那我们简单来说说这个final
:
- final修饰成员变量时,该成员变量自定义起就不能被修改
- 当该成员变量为基本类型时,初始化赋值后,值不能被修改
- 当该成员变量为引用类型时,初始化对象后,不能重新分配内存空间
- final修饰方法时,该方法为最终方法,不能被重写
- final修饰类时,该类就为最终类,不能被继承
- 也就是说,String类被定义成最终类,不允许有子类继承
但是在源码中都是这么实现的吗??? 不,不,不,classMath
实现方案就不同。
从源码角度来看的话,此类不仅不能被继承,没有默认构造方法,现有的构造方法被private
修饰。所以此类是不能被实例化的。
所以当我们使用Math
中的方法的时候,都是通过静态方法来调用。
2. 应用
其实不能被继承的方法有很多,都是Java定义好提供给我们的。例如:
- Math
- String
- Integer
那为什么要用final
修饰呢
-
安全性:
- String类是JDK提供的核心类之一,在String类中有许多方法都是本地方法,在使用这类方法时,需要频繁地和操作系统进行交流访问,如果不定义为final,允许被继承和重写String类中的方法,那么就容易出现被恶意攻击的现象
-
字符串常量池的需要
- 为了节省空间,java引出了常量池的概念,正是因为字符串的不可变,常量池才有可能实现。如果字符串是可变的,那对于常量池中一个字符串的改变,也会引起其他指向该字符串的字符串对象发生变化,这样一来字符串常量池就实现不了
-
hashcode值
- 当字符串被创建时,hashcode值就被计算出来并缓存,想获取该字符串的hashcode 时,只需要从常量池中找到对应字符串再获取对应值即可,如果String是可变的,那每创建一个String实例,hashcode值就需要再计算一次,这样就使得使效率大大降低。也正是因为String的不可变,才使得大多数map中建的类型都是用String存储(map中需要根据hashcode值来计算存储位置,String的hashcode不可变处理速度很快)
-
多线程安全
- 由于被定义为不可变的缘故,所以String的多线程是安全的,同一个字符串对象可以同时被多个对象共享