前言
有时候一些类只提供静态方法,通常是一些工具类,这些类不希望被实例化,那么我们就可以通过私有构造器来强化不可实例化的能力
通过私有构造器强化不可实例化的能力
1. 介绍
有些类只需要包含静态方法和静态域,比如 java.lang.Math 或者 java.util.Arrays,把基本类型的值或者数组类型上的相关方法组织起来。我们也可以通过 java.util.Collections 的方式将实现特定接口的对象上的静态方法,包括工厂方法组织起来。从 java8 开始,也可以把这些方法放入接口中,假设这是你自己编写的接口可以进行修改。总之一句话就是把同种功能的常量或者方法存到一个类中,这种类通常就是只包含静态方法或静态域的,我们都希望直接通过类调用而不是 new 一个类出来。
2. 解决
这种情况下实例化对这些工具类是没有意义的,然而默认情况下,编译器会自动提供一个公有的、无参的缺省构造器(default constructor)。对于用户而言,这个构造器和其他构造器没有区别,在某些情况下可能就会被实例化了。那么解决这个问题,我们可以在私有构造器里面抛出异常:
public class UtilityClass {
private static final UtilityClass utilityClass = new UtilityClass();
//防止内部实例化
private UtilityClass(){
throw new AssertionError();
}
public static UtilityClass getInstance(){
return utilityClass;
}
}
私有的构造器使得不能在外部访问它,而构造器抛出异常可以避免内部访问,当用了这种写法的时候,应该加一行注释告诉调用者为什么要这么做。
3. 其他
当然了,我们不能通过把这个类变成抽象类来避免被实例化,因为该类可以被子类化,字类可以被实例化,这样可能会误导用户这个类是为了继承而设置的,反而有可能提醒了用户这个类是需要实例化的。而私有构造器的写法也是有不好的地方的,它使得一个类不能被实例化,所有的构造器都必须隐式调用超类(superclass,也就是父类)构造器,在这种情况下,子类就没有可以访问的超类构造器可以调用了。
如有错误,欢迎指出!!!!