有两种常见的方式来实现单例。 两者都基于保持构造方法私有和导出公共静态成员以提供对唯一实例访问
第一种
// Singleton with public final field
public class Elvis {
// 赋初值 定义为final属性
public static final Elvis INSTANCE = new Elvis();
private Elvis() { ... }
public void leaveTheBuilding() { ... }
第二种
// Singleton with static factory
public class Elvis {
private static final Elvis INSTANCE = new Elvis();
private Elvis() { ... }
// 公共成员为静态的工厂方法,限制实例的创建
public static Elvis getInstance() { return INSTANCE; }
public void leaveTheBuilding() { ... }
}
如果要想保证单例在序列化的情况下不被破坏,需要将字段声明为transient,并重写readreslve方法:
// readResolve method to preserve singleton property
private Object readResolve() {
// Return the one true Elvis and let the garbage collector
// take care of the Elvis impersonator.
return INSTANCE;
}
这两种方式可以通过反射,直接调用私有构造的方式破坏掉。
第三种方法是声明一个单一元素的枚举类:
// Enum singleton - the preferred approach
public enum Elvis {
INSTANCE;
public void leaveTheBuilding() { ... }
}
枚举类型,更简洁,提供无偿的序列化机制,并提供了防止多个实例化的坚固保证,但不满足单例需要继承其他类的应用场景。
当编写一个工具类,提供一些静态方法与字段,不想被实例化与继承时,也可以将构造方法私有化,防止类被实例化
本文介绍了两种常见的Java单例实现方法,包括私有构造方法和枚举方式。私有构造方法结合公共静态成员确保唯一实例,但可被反射破坏。枚举方式则提供更安全的序列化保护,且无法被继承。对于不希望被实例化和继承的工具类,可以使用私有构造方法。
7990

被折叠的 条评论
为什么被折叠?



