【面试题Java】单例模式

单例模式的实现方法

  • 饿汉式:在类加载时就立即创建单例实例,线程安全,实现简单。代码示例如下:
java 复制代码
public class Singleton {
    // 私有静态实例,在类加载时就创建
    private static final Singleton instance = new Singleton();

    // 私有构造函数,防止外部实例化
    private Singleton() {}

    // 公共的静态方法,用于获取单例实例
    public static Singleton getInstance() {
        return instance;
    }
}
  • 懒汉式(线程不安全) :在第一次调用getInstance方法时才创建实例,实现较为简单,但在多线程环境下可能会创建多个实例。代码示例如下:
java 复制代码
public class Singleton {
    // 私有静态实例,初始化为null
    private static Singleton instance = null;

    // 私有构造函数,防止外部实例化
    private Singleton() {}

    // 公共的静态方法,用于获取单例实例
    public static Singleton getInstance() {
        if (instance == null) {
            instance = new Singleton();
        }
        return instance;
    }
}
  • 懒汉式(线程安全) :通过synchronized关键字保证在多线程环境下只有一个线程能创建实例,但性能可能会受影响。代码示例如下:
java 复制代码
public class Singleton {
    // 私有静态实例,初始化为null
    private static Singleton instance = null;

    // 私有构造函数,防止外部实例化
    private Singleton() {}

    // 公共的静态方法,用于获取单例实例,添加synchronized关键字保证线程安全
    public static synchronized Singleton getInstance() {
        if (instance == null) {
            instance = new Singleton();
        }
        return instance;
    }
}
  • 双重检查锁(DCL) :在getInstance方法中进行两次null检查,提高了性能,同时保证线程安全。代码示例如下:
java 复制代码
public class Singleton {
    // 私有静态实例,使用volatile关键字保证可见性和禁止指令重排序
    private static volatile Singleton instance = null;

    // 私有构造函数,防止外部实例化
    private Singleton() {}

    // 公共的静态方法,用于获取单例实例
    public static Singleton getInstance() {
        if (instance == null) {
            synchronized (Singleton.class) {
                if (instance == null) {
                    instance = new Singleton();
                }
            }
        }
        return instance;
    }
}
  • 静态内部类 :利用静态内部类的特性,在外部类加载时,静态内部类不会被加载,只有在调用getInstance方法时才会加载内部类并创建实例,线程安全且性能较好。代码示例如下:
java 复制代码
public class Singleton {
    // 私有构造函数,防止外部实例化
    private Singleton() {}

    // 静态内部类,用于持有单例实例
    private static class SingletonHolder {
        // 私有静态实例,在类加载时创建
        private static final Singleton instance = new Singleton();
    }

    // 公共的静态方法,用于获取单例实例
    public static Singleton getInstance() {
        return SingletonHolder.instance;
    }
}
  • 枚举:通过枚举类型实现单例,简洁且天然支持序列化和线程安全。代码示例如下:
java 复制代码
public enum Singleton {
    // 唯一的实例
    INSTANCE;

    // 可以添加其他方法和属性
    public void doSomething() {
        // 具体实现
    }
}

破坏单例的方式

  • 反射:通过反射机制可以调用私有构造函数来创建新的实例,从而破坏单例。代码示例如下:
java 复制代码
import java.lang.reflect.Constructor;

public class Main {
    public static void main(String[] args) throws Exception {
        // 获取单例类的Class对象
        Class<Singleton> clazz = Singleton.class;
        // 获取私有构造函数
        Constructor<Singleton> constructor = clazz.getDeclaredConstructor();
        // 允许访问私有构造函数
        constructor.setAccessible(true);
        // 通过构造函数创建新的实例
        Singleton instance1 = constructor.newInstance();
        // 获取单例实例
        Singleton instance2 = Singleton.getInstance();
        // 比较两个实例是否相同
        System.out.println(instance1 == instance2); 
    }
}
  • 反序列化 :如果单例类实现了Serializable接口,在反序列化时会创建一个新的实例,从而破坏单例。代码示例如下:
java 复制代码
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.Serializable;

public class Main {
    public static void main(String[] args) throws Exception {
        // 创建单例实例
        Singleton instance1 = Singleton.getInstance();
        // 将单例实例序列化到文件
        try (ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("singleton.obj"))) {
            oos.writeObject(instance1);
        }
        // 从文件中反序列化出单例实例
        try (ObjectInputStream ois = new ObjectInputStream(new FileInputStream("singleton.obj"))) {
            Singleton instance2 = (Singleton) ois.readObject();
            // 比较两个实例是否相同
            System.out.println(instance1 == instance2); 
        }
    }
}

class Singleton implements Serializable {
    // 私有静态实例
    private static final Singleton instance = new Singleton();

    // 私有构造函数
    private Singleton() {}

    // 公共的静态方法,用于获取单例实例
    public static Singleton getInstance() {
        return instance;
    }

    // 反序列化时,返回单例实例
    private Object readResolve() {
        return instance;
    }
}
  • 对象克隆 :如果单例类实现了Cloneable接口,并且没有正确处理克隆方法,通过克隆也可以创建新的实例,从而破坏单例。代码示例如下:
java 复制代码
public class Main {
    public static void main(String[] args) throws Exception {
        // 获取单例实例
        Singleton instance1 = Singleton.getInstance();
        // 克隆单例实例
        Singleton instance2 = (Singleton) instance1.clone();
        // 比较两个实例是否相同
        System.out.println(instance1 == instance2); 
    }
}

class Singleton implements Cloneable {
    // 私有静态实例
    private static final Singleton instance = new Singleton();

    // 私有构造函数
    private Singleton() {}

    // 公共的静态方法,用于获取单例实例
    public static Singleton getInstance() {
        return instance;
    }

    @Override
    protected Object clone() throws CloneNotSupportedException {
        // 如果不重写clone方法,默认会创建一个新的实例
        return super.clone();
    }
}

为了防止反射和反序列化破坏单例,可以在单例类的私有构造函数中添加逻辑,判断是否已经创建过实例,如果是则抛出异常。对于对象克隆,可以重写clone方法,返回单例实例而不是创建新的实例。对于枚举类型的单例,由于其本身的特性,天然防止了反射、反序列化和对象克隆的破坏。

相关推荐
Forget the Dream1 小时前
设计模式之迭代器模式
java·c++·设计模式·迭代器模式
周努力.1 小时前
设计模式之单例模式
单例模式·设计模式
咩咩觉主1 小时前
C# &Unity 唐老狮 No.7 模拟面试题
开发语言·unity·c#
大丈夫在世当日食一鲲1 小时前
Java中用到的设计模式
java·开发语言·设计模式
A-Kamen1 小时前
Spring Boot拦截器(Interceptor)与过滤器(Filter)深度解析:区别、实现与实战指南
java·spring boot·后端
练川1 小时前
Stream特性(踩坑):惰性执行、不修改原始数据源
java·stream
狂奔小菜鸡1 小时前
Java运行时数据区
java·jvm·后端
trymoLiu1 小时前
SpringBoot 实现 RSA+AES 自动接口解密!
java·spring boot
却道天凉_好个秋2 小时前
c++ 嵌入汇编的方式实现int型自增
开发语言·汇编·c++
ChinaRainbowSea2 小时前
MySQL 索引的数据结构(详细说明)
java·数据结构·数据库·后端·mysql