Effective Java第4条:通过私有构造器强化不可实例化的能力(精简与解析)

通过私有构造器强化不可实例化的能力

在面向对象编程中,有时我们需要创建一些只包含静态方法和静态字段的工具类(Utility Class),这些类不应该被实例化。

有些类只是作为静态方法和静态字段的容器,它们不需要也不应该被实例化。
比如:

  • 数学工具类(MathUtils)
  • 字符串处理工具类(StringUtils)
  • 常量定义类(Constants)
  • 日志工具类(Logger)
  1. 在没有显示的编写构造类时,编译器就会提供一个共有、无参的缺省构造器(默认)。于用户而言,这个构造器与其他的构造器没有任何区别。可能会被无意识地实例化。

  2. 企图通过将类做成抽象类来强制该类不可被实例化是行不通的,该类可以子类化并且被实例化,会误导用户是专门为了继承而设计。

由于只有当类不包含显式的构造器时,编译器才会生成缺省的构造器,因此只要让这个类包含一个私有构造器就可以。

// 不可实例化的工具类
public class StringUtils {
    // 私有构造器
    private StringUtils() {
        throw new AssertionError("Cannot be instantiated");
    }
    
    public static boolean isEmpty(String str) {
        return str == null || str.length() == 0;
    }
}

// 尝试继承 - 这会编译错误!
public class ExtendedStringUtils extends StringUtils {  // 编译错误
    // 子类构造器必须调用父类构造器(显式或隐式)
    // 但父类只有私有构造器,无法访问
}

ssertiongrror不是必需的,但是它可以避免不小心在类的内部调用构造器。它保证该类在任何情况下都不会被实例化。这种习惯用法有点违背直觉,好像构造器就是专门设计成不能被调用一样。因此,明智的做法就是在代码中增加一条注释。

不过这样就此类就没办法子类化了。子类构造器必须调用父类构造器(显式或隐式),但父类只有私有构造器,无法访问。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

Cosmoshhhyyy

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值