《Effective Java 中文版 第二版》第二章 第4条:通过私有构造器强化不可实例化的能力

本章的主题是创建和销毁对象:何时以及如何创建对象,何时以及如何避免创建对象,如何确保它们能够适时地销毁,以及如何管理对象销毁之前必须进行的各种清理动作。


[toc]

有时候,你可能需要编写只包含静态方法和静态域的类。虽然这些类名声很不好,因为有些人在面向对象的语言中滥用这样的类来编写过程化的程序。尽管如此,它们也确实有它们特有的用处。我们可以利用这种类,以 java.lang.Math 或者 java.util.Arrays 的方式,把基本类型的值或者数组类型上的相关方法组织起来。我们也可以通过 java.util.Collections 的方式,把实现特定接口的对象上的静态方法(包括静态方法,见第1条)组织起来。最后还可以利用这种类把 final 类上的方法组织起来,以取代扩展该类的做法。

这样的工具类(utility class)不希望被实例化,实例对它没有任何意义。然而在缺少显石子构造器的情况下,编译器会自动提供一个公有的、无参的缺省构造器(default constructor)。对于用户而言,这个构造器与其他构造器没有任何区别。在已经发行的API中常常可以看到一些被无意识地实例化的类。

企图通过将类做成抽象类来强制该类不可被实例化,这是行不通的
该类可以被子类化,并且该子类也可以被实例化。这样可能会误导用户,以为这种类是专门为了继承而设计的(见 17 条)。

而我们只需要让这个类只包含私有构造器,它就不能被实例化了:

// Noninstantiable utility class
public class UtilityClass {
    // Suppress default constructor for noninstantiability
    private UtilityClass() {
        throw new AssertionError();
    }
    ... // Remainder omitted
}

由于显式的构造器是私有的,所以不可以在该类的外部访问它。AssertionError 不是必须的,它可以避免不小心在类的内部调用构造器,防止该类被实例化,这种习惯用法有点违背直觉,好像构造器就是专门设计成不能被调用一样。所以最好加上一条注释。如上所示。

副作用:

  • 使得一个类不能被子类化。
    所有的构造器都必须显式或隐式地调用超类(superclass)构造器,在这种情况下,子类就没有可访问的超类构造器可调用了。
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 32,483评论 18 399
  • 一:java概述:1,JDK:Java Development Kit,java的开发和运行环境,java的开发工...
    ZaneInTheSun阅读 7,618评论 0 11
  • 本章将会介绍 存储属性的初始赋值自定义构造过程默认构造器值类型的构造器代理类的继承和构造过程可失败构造器必要构造器...
    寒桥阅读 4,134评论 0 0
  • 因为骨子里很现实,担忧自己变得现实,所以才会追求理想吧 被人说是理想主义者,好像也没有什么不好的,现实真是让人哭笑...
    ALeesa阅读 1,499评论 0 0
  • 请认真思考一下,你的公司是不是存在着这样的普遍现象—— 有5%~10%的公司员工,一上班就是来挑毛病、和你对着干,...
    小声讲故事阅读 2,241评论 0 0

友情链接更多精彩内容