Java单例类与对象序列化
为了保证全局只有一个实例,单例类往往将构造方法的访问权限设置为private:
public class Singleton implements Serializable{ private static final Singleton SINGLETON = new Singleton(100); private int value; private Singleton(int value) { // private! this.value = value; } public static Singleton getInstance() { return SINGLETON; } }
对单例类进行序列化需要格外小心,下面看一段代码:
public static void main(String[] args) throws IOException, ClassNotFoundException { Singleton singleton1 = Singleton.getInstance(); ObjectOutputStream output = new ObjectOutputStream(new FileOutputStream("singleton.dat")); output.writeObject(singleton1); ObjectInputStream input = new ObjectInputStream(new FileInputStream("singleton.dat")); Singleton singleton2 = (Singleton) input.readObject(); System.out.println("创建新实例:" + (singleton1 != singleton2)); }
控制台输出:
创建新实例:true
上面代码的运行结果表明,对单例类进行序列化可能会创建新的实例,即使构造方法是私有的!
为了解决上述问题,需要定义一种被称为readResolve的特殊序列化方法。如果定义了readResolve方法,在对象序列化时就会调用它。
public class Singleton implements Serializable { private static final Singleton SINGLETON = new Singleton(100); private int value; private Singleton(int value) { // private! this.value = value; } public static Singleton getInstance() { return SINGLETON; } protected Object readResolve() throws ObjectStreamException { return SINGLETON; } }
再次执行上面的测试代码,控制台输出如下:
创建新实例:false
地势坤,君子以厚德载物。
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】凌霞软件回馈社区,博客园 & 1Panel & Halo 联合会员上线
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】博客园社区专享云产品让利特惠,阿里云新客6.5折上折
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步