“Github:https://github.com/nateshao/design-demo/tree/main/JavaDesignPatterns/08-singleton
如何保证一个类只有一个实例并且这个实例易于被访问?
单例模式的定义
“单例模式:确保一个类只有一个实例,并提供一个全局访问点来访问这个唯一实例。
对象创建型模式
要点:
单例模式的结构
**单例模式只包含一个单例角色:**Singleton(单例)
单例模式的实现
public class Singleton {
private static Singleton instance = null; //静态私有成员变量
//私有构造函数
private Singleton() {
}
//静态公有工厂方法,返回唯一实例
public static Singleton getInstance() {
if (instance == null)
instance = new Singleton();
return instance;
}
}
实例说明:
某软件公司承接了一个服务器负载均衡(Load Balance)软件的开发工作,该软件运行在一台负载均衡服务器上,可以将并发访问和数据流量分发到服务器集群中的多台设备上进行并发处理,提高了系统的整体处理能力,缩短了响应时间。由于集群中的服务器需要动态删减,且客户端请求需要统一分发,因此需要确保负载均衡器的唯一性,只能有一个负载均衡器来负责服务器的管理和请求的分发,否则将会带来服务器状态的不一致以及请求分配冲突等问题。如何确保负载均衡器的唯一性是该软件成功的关键,试使用单例模式设计服务器负载均衡器。
实例类图
实例代码
分析:
//判断服务器负载均衡器是否相同
if (balancer1 == balancer2 && balancer2 == balancer3 && balancer3 == balancer4) {
System.out.println("服务器负载均衡器具有唯一性!");
}
饿汉式单例类(Eager Singleton)
public class EagerSingleton {
private static final EagerSingleton instance = new EagerSingleton();
private EagerSingleton() {
}
public static EagerSingleton getInstance() {
return instance;
}
}
懒汉式单例类(Lazy Singleton)
延迟加载
public class LazySingleton {
private static LazySingleton instance = null;
private LazySingleton() { }
public static LazySingleton getInstance() {
if (instance == null) {
instance = new LazySingleton();
}
return instance;
}
}
多个线程同时访问将导致创建多个单例对象!怎么办?
public class LazySingleton {
private static LazySingleton instance = null;
private LazySingleton() { }
// 锁方法
synchronized public static LazySingleton getInstance() {
if (instance == null) {
instance = new LazySingleton();
}
return instance;
}
}
……
public static LazySingleton getInstance() {
if (instance == null) {
// 锁代码段
synchronized (LazySingleton.class) {
instance = new LazySingleton();
}
}
return instance;
}
……
public class LazySingleton {
private volatile static LazySingleton instance = null;
private LazySingleton() { }
// Double-Check Locking 双重检查锁定
public static LazySingleton getInstance() {
//第一重判断
if (instance == null) {
//锁定代码块
synchronized (LazySingleton.class) {
//第二重判断
if (instance == null) {
instance = new LazySingleton(); //创建单例实例
}
}
}
return instance;
}
}
饿汉式单例类:无须考虑多个线程同时访问的问题;调用速度和反应时间优于懒汉式单例;资源利用效率不及懒汉式单例;系统加载时间可能会比较长
懒汉式单例类:实现了延迟加载;必须处理好多个线程同时访问的问题;需通过双重检查锁定等机制进行控制,将导致系统性能受到一定影响
//Initialization on Demand Holder
public class Singleton {
private Singleton() {
}
//静态内部类
private static class HolderClass {
private final static Singleton instance = new Singleton();
}
public static Singleton getInstance() {
return HolderClass.instance;
}
public static void main(String args[]) {
Singleton s1, s2;
s1 = Singleton.getInstance();
s2 = Singleton.getInstance();
System.out.println(s1==s2);
}
}
模式优点
模式缺点
模式适用环境