【面试题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方法,返回单例实例而不是创建新的实例。对于枚举类型的单例,由于其本身的特性,天然防止了反射、反序列化和对象克隆的破坏。

相关推荐
bing_1583 分钟前
Java 中求两个 List集合的交集元素
java·list
장숙혜5 分钟前
JavaScript正则表达式解析:模式、方法与实战案例
开发语言·javascript·正则表达式
工业互联网专业22 分钟前
基于springboot+vue的高校社团管理系统的设计与实现
java·vue.js·spring boot·毕业设计·源码·课程设计
安大小万22 分钟前
C++ 学习:深入理解 Linux 系统中的冯诺依曼架构
linux·开发语言·c++
九圣残炎24 分钟前
【ElasticSearch】 Java API Client 7.17文档
java·elasticsearch·搜索引擎
随心Coding26 分钟前
【零基础入门Go语言】错误处理:如何更优雅地处理程序异常和错误
开发语言·后端·golang
T.Ree.31 分钟前
C语言_自定义类型(结构体,枚举,联合)
c语言·开发语言
Channing Lewis32 分钟前
python生成随机字符串
服务器·开发语言·python
小熊科研路(同名GZH)1 小时前
【Matlab高端绘图SCI绘图模板】第002期 绘制面积图
开发语言·matlab