一、什么是单例设计模式
概念:
java中单例模式是一种常见的设计模式,单例模式的写法有好几种,这里主要介绍俩种:懒汉式单例、饿汉式单例。
单例模式有以下特点:
1、单例类只能有一个实例。
2、单例类必须自己创建自己的唯一实例。
3、单例类必须给所有其他对象提供这一实例。
单例模式确保某个类只有一个实例,而且自行实例化并向整个系统提供这个实例。在计算机系统中,线程池、缓存、日志对象、对话框、打印机、显卡的驱动程序对象常被设计成单例。这些应用都或多或少具有资源管理器的功能。每台计算机可以有若干个打印机,但只能有一个Printer Spooler,以避免两个打印作业同时输出到打印机中。每台计算机可以有若干通信端口,系统应当集中管理这些通信端口,以避免一个通信端口同时被两个请求同时调用。总之,选择单例模式就是为了避免不一致状态,避免政出多头。
1.饿汉式单例
// 单例设计模式,饿汉式
public class Single {
private Single() {
}
private static Single single = new Single();
public static Single getSingle() {
return single;
}
}
类在外部创建不了对象,可以内部创建,有两种形式,一种是属性创建,一种是方法返回 但是由于外部没有对象,所以两种方式都需要static修饰,从而通过类名.的形式调用 用static修饰的原因是,静态方法中不能直接调用非静态的属性跟方法 属性私有化的原因是不能让其在外部直接获取属性,虽然不能创建新的对象,但是可以销毁已有的对象,因此属性私有化
2.懒汉式单例
//单例设计模式,懒汉式
public class Singles {
private static Singles singles = null;
private Singles(){}
public static Singles getSingles(){
if (singles == null){
singles = new Singles();
}
return singles;
}
}
Singles通过将构造方法限定为private避免了类在外部被实例化,在同一个虚拟机范围内,Singles的唯一实例只能通过getSingles()方法访问。
二、饿汉式与懒汉式的区别
饿汉式和懒汉式是单例模式中两种不同的实例化方式,它们的主要区别在于实例的创建时机。
饿汉式
在饿汉式单例模式中,实例在类加载时就被创建,而不考虑是否需要这个实例。这种方式的优点是实现简单,线程安全,因为实例在类加载时就创建好了,所以不会有线程安全问题。缺点是如果这个实例从未被使用,那么创建实例的资源就被浪费了。
懒汉式
在懒汉式单例模式中,实例在第一次被请求时才被创建。这种方式的优点是延迟了实例的创建,节省了资源,只有当实例真正需要时才被创建。缺点是实现稍微复杂一些,并且需要考虑线程安全问题。
三、双重检查锁定
为了保证线程安全,可以使用双重检查锁定(Double-Checked Locking)或者使用 synchronized
关键字。
public class LazySingleton {
// 声明实例,但不初始化
private static volatile LazySingleton instance;
// 私有构造函数,防止外部实例化
private LazySingleton() {
}
// 提供一个全局访问点
public static LazySingleton getInstance() {
if (instance == null) {
synchronized (LazySingleton.class) {
if (instance == null) {
instance = new LazySingleton();
}
}
}
return instance;
}
}
在这个线程安全的版本中,使用 volatile
关键字确保 instance
变量的可见性,并且使用双重检查锁定来减少同步的开销。
四、总结
饿汉式:实例在类加载时创建,线程安全,但可能浪费资源。
懒汉式:实例在第一次使用时创建,节省资源,但需要考虑线程安全问题。
选择哪种方式取决于具体的需求和应用场景。如果实例的创建开销不大,且一定会被使用,可以选择饿汉式;如果实例的创建开销较大,或者不一定被使用,可以选择懒汉式,并确保线程安全。