了解单例模式之前先理解一个概念:延时加载
延时加载是指只有当对象在真正使用的时候才会被加载到内存中创建实例,不使用时不会创建。从速度和反应时间角度来讲,非延迟加载(即饿汉式)好,从资源利用率上说,延迟加载(又称懒汉式)好
单例模式的实现方式有五种,如下所示:
主要:
- 饿汉式 - 线程安全,调用效率高,但是不支持延迟加载。
- 懒汉式 - 线程安全,调用效率不高,可以支持延时加载。
其他:
- 双重检测锁式(由于JVM底层内部模型原因,偶尔会出问题,不建议使用)
- 静态内部类式 (线程安全,调用效率高,而且,可以支持延时加载)
- 枚举式 (线程安全,调用效率高,不能延时加载,并且可以天然的防止反射和反序列化漏洞!)
如何选用?
单例对象 占用资源少, 不需要延时加载:
- 枚举式 好过 饿汉式
单例对象, 占用资源大,需要延时加载:
- 静态内部类式 好过 懒汉式
饿汉式- 代码示例
public class SingtonInstance1 {
private static SingtonInstance1 instance = new SingtonInstance1();
private SingtonInstance1() {
}
public static SingtonInstance1 getInstance() {
return instance;
}
}
懒汉式 - 代码示例
public class SingtonInstance2 {
private static SingtonInstance2 instance ;
private SingtonInstance2() {
}
public static synchronized SingtonInstance2 getInstance() {
if(instance == null) {
instance = new SingtonInstance2();
}
return instance;
}
}
静态内部类式 - 代码示例
public class SingtonInstance3 {
private static class SingtonInnerInstance{
private static final SingtonInstance3 instance = new SingtonInstance3();
}
private SingtonInstance3() {
}
public static SingtonInstance3 getInstance() {
return SingtonInnerInstance.instance;
}
}
枚举式 - 代码示例
package com.begin.utils;
public enum SingtonInstance4 {
INSTANCE;
public void operation() {
}
}
接下来一个知识点的补充:
五种单例模式中只有枚举式是一个天然的防止反射和反序列化漏洞的单例模式。如何让其他几种也可以防止反射和反序列化漏洞呢。代码如下所示:(一般情况,不需要考虑这些,除非你要重写一个jdk)
防止通过反射破解单例: 可以在私有的构造函数中添加如下代码:
private SingtonInstance1() {
if(instance !=null) {
throw new RuntimeException();
}
}
防止反序列化的方法:
//反序列化,如果定义了readResolve方法则直接返回此方法指定的对象。而不需要单独创建对象
private Object readResolve() throws ObjectStreamException{
return instance;
}