浅谈设计模式 - 单例模式(一)(下)

简介: 浅谈设计模式 - 单例模式(一)(下)

静态代码块



和饿汉式差不多,这里不在过多赘述,直接上代码:


实现代码:


/**
 * 静态代码块的形式,实现单例
 *
 * @Author zhaoxudong
 * @Date 2020/10/28 13:28
 **/
public class StaticBlock {
    private static final StaticBlock staticBlock;
    static {
        staticBlock = new StaticBlock();
    }
    public static StaticBlock getInstance() {
        return staticBlock;
    }
}
复制代码


静态内部类:



优点:


  1. 既可以保证一次加载,又可以保证不出现重复的初始化
  2. 可以用一个大类管理所有的内部类


缺点:


  1. 额外需要多一个内部类
  2. 破坏代码设计模式


实现代码:


package com.zxd.interview.desginpattern.single;
import com.zxd.interview.util.ExecuteUtil;
/**
 * 单例模式 - 静态内部类实现
 *
 * @Author zhaoxudong
 * @Date 2020/10/28 13:35
 **/
public class SingleStaticInner {
    /**
     * 使用内部类来进行后续的构造
     */
    public static class Instatnce {
        private static Instatnce instatnce = new Instatnce();
        public static Instatnce getInstatnce() {
            try {
                // 模拟在创建对象之前做一些准备工作
                Thread.sleep(1000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
            return instatnce;
        }
    }
    public static void main(String[] args) throws InterruptedException {
        ExecuteUtil.startTaskAllInOnce(250, new ThreadTest());
    }
}
/**
 * 测试多线程获取对象
 */
class ThreadTest extends Thread {
    @Override
    public void run() {
        System.err.println(SingleStaticInner.Instatnce.getInstatnce());
    }
}
复制代码


序列化/反序列化的问题:



解释:序列化和反序列化的情况下,会出现问题,因为JAVA的序列化从磁盘读取的时候,会生成新的实例对象,但是这样就会违背单例模式的方式


实现代码:


package com.zxd.interview.desginpattern.single;
import java.io.*;
/**
 * 单例模式 - 序列化与反序列化的问题和解决办法
 * @Author zhaoxudong
 * @Date 2020/10/28 13:55
 **/
public class SingleSerialize {
    public static void main(String[] args) throws IOException, ClassNotFoundException {
        SerializeStaticInner instance = SerializeStaticInner.getInstance();
        System.err.println(instance.hashCode());
        // 序列化
        FileOutputStream fileOutputStream = new FileOutputStream("temp");
        ObjectOutputStream objectOutputStream = new ObjectOutputStream(fileOutputStream);
        objectOutputStream.writeObject(instance);
        objectOutputStream.close();
        fileOutputStream.close();
        // 反序列化
        FileInputStream fileInputStream = new FileInputStream("temp");
        ObjectInputStream objectInputStream = new ObjectInputStream(fileInputStream);
        SerializeStaticInner read = (SerializeStaticInner) objectInputStream.readObject();
        objectInputStream.close();
        fileInputStream.close();
        System.err.println(read.hashCode());
    }
    static class SerializeStaticInner implements Serializable{
        private static  SerializeStaticInner serializeStaticInner = new SerializeStaticInner();
        public static SerializeStaticInner getInstance(){
            return serializeStaticInner;
        }
        /**
         * 序列化当中的一个钩子方法
         * 避免序列化和反序列化的对象为新实例破坏单例模式的规则
         */
//        protected Object readResolve(){
//            System.err.println("调用特定的序列化方法");
//            return SerializeStaticInner.serializeStaticInner;
//        }
    }
}
复制代码


  1. 如果没有readResolve(),那么序列化之后反序列化是会变为一个新的实例,这样会破坏单例模式
  2. 如果存在readResolve(),那么序列化之后的对象就不会出现多个实例


扩展:为什么加入readResolve() 方法就可以避免序列化的问题


下面是关于《effective Java》的解释


网络异常,图片无法展示
|

关于此方法的访问权注意事项

网络异常,图片无法展示
|


扩展:序列化必知:



  • 所有需要网络传输的对象都需要实现序列化接口,通过建议所有的javaBean都实现Serializable接口。
  • 对象的类名、实例变量(包括基本类型,数组,对其他对象的引用)都会被序列化;方法、类变量、transient实例变量都不会被序列化。
  • 如果想让某个变量不被序列化,使用transient修饰。
  • 序列化对象的引用类型成员变量,也必须是可序列化的,否则,会报错。
  • 反序列化时必须有序列化对象的class文件。
  • 当通过文件、网络来读取序列化后的对象时,必须按照实际写入的顺序读取。
  • 单例类序列化,需要重写readResolve()方法;否则会破坏单例原则。
  • 同一对象序列化多次,只有第一次序列化为二进制流,以后都只是保存序列化编号,不会重复序列化。
  • 建议所有可序列化的类加上serialVersionUID 版本号,方便项目升级。
相关文章
|
6月前
|
设计模式 缓存 安全
【设计模式】【创建型模式】单例模式(Singleton)
一、入门 什么是单例模式? 单例模式是一种设计模式,确保一个类只有一个实例,并提供一个全局访问点。它常用于需要全局唯一对象的场景,如配置管理、连接池等。 为什么要单例模式? 节省资源 场景:某些对象创
247 15
|
设计模式 安全 Java
Kotlin教程笔记(57) - 改良设计模式 - 单例模式
Kotlin教程笔记(57) - 改良设计模式 - 单例模式
112 2
|
8月前
|
设计模式 安全 Java
设计模式:单例模式
单例模式是一种创建型设计模式,确保一个类只有一个实例,并提供全局访问点。它通过私有化构造函数、自行创建实例和静态方法(如`getInstance()`)实现。适用于数据库连接池、日志管理器等需要全局唯一对象的场景。常见的实现方式包括饿汉式、懒汉式、双重检查锁、静态内部类和枚举。线程安全问题可通过`synchronized`或双重检查锁解决,同时需防止反射和序列化破坏单例。优点是避免资源浪费,缺点是可能增加代码耦合度和测试难度。实际开发中应优先选择枚举或静态内部类,避免滥用单例,并结合依赖注入框架优化使用。
|
7月前
|
设计模式 存储 安全
设计模式-单例模式练习
单例模式是Java设计模式中的重要概念,确保一个类只有一个实例并提供全局访问点。本文详解单例模式的核心思想、实现方式及线程安全问题,包括基础实现(双重检查锁)、懒汉式与饿汉式对比,以及枚举实现的优势。通过代码示例和类图,深入探讨不同场景下的单例应用,如线程安全、防止反射攻击和序列化破坏等,展示枚举实现的简洁与可靠性。
139 0
|
9月前
|
设计模式 存储 安全
设计模式2:单例模式
单例模式是一种创建型模式,确保一个类只有一个实例,并提供全局访问点。分为懒汉式和饿汉式: - **懒汉式**:延迟加载,首次调用时创建实例,线程安全通过双重检查锁(double check locking)实现,使用`volatile`防止指令重排序。 - **饿汉式**:类加载时即创建实例,线程安全但可能浪费内存。 示例代码展示了如何使用Java实现这两种模式。
230 4
|
11月前
|
设计模式 存储 前端开发
前端必须掌握的设计模式——单例模式
单例模式是一种简单的创建型设计模式,确保一个类只有一个实例,并提供一个全局访问点。适用于窗口对象、登录弹窗等场景,优点包括易于维护、访问和低消耗,但也有安全隐患、可能形成巨石对象及扩展性差等缺点。文中展示了JavaScript和TypeScript的实现方法。
483 13
|
11月前
|
设计模式 安全 Java
Kotlin教程笔记(57) - 改良设计模式 - 单例模式
Kotlin教程笔记(57) - 改良设计模式 - 单例模式
162 2
|
设计模式 Java 数据库连接
Java编程中的设计模式:单例模式的深度剖析
【10月更文挑战第41天】本文深入探讨了Java中广泛使用的单例设计模式,旨在通过简明扼要的语言和实际示例,帮助读者理解其核心原理和应用。文章将介绍单例模式的重要性、实现方式以及在实际应用中如何优雅地处理多线程问题。
199 4
|
设计模式 安全 Java
Kotlin教程笔记(57) - 改良设计模式 - 单例模式
Kotlin教程笔记(57) - 改良设计模式 - 单例模式
|
设计模式 存储 数据库连接
PHP中的设计模式:单例模式的深入理解与应用
【10月更文挑战第22天】 在软件开发中,设计模式是解决特定问题的通用解决方案。本文将通过通俗易懂的语言和实例,深入探讨PHP中单例模式的概念、实现方法及其在实际开发中的应用,帮助读者更好地理解和运用这一重要的设计模式。
131 1