思考问题:如果你希望整个项目中,某个类只能有一个实例,怎么设计?
单例模式有两种表现形式:
饿汉式单例类
(类加载时直接初始化实例,所以饥饿) 实例代码:public class Singleton {
//在类加载时生成实例
private static Singleton instance = new Singleton();
//私有化构造函数
private Singleton(){}
//通过该方法获得实例引用
public static Singleton getInstance() {
return instance;
}
}
注意点:
- 实例引用必须声明为static才能在类加载时初始化。
- 必须显式声明private构造函数,避免该单例类被其他类直接new对象、也避免源码编译时被添加默认构造函数。(因为类没有显式声明构造函数时,编译器在编译时会加上无参构造函数)
- getInstance函数必须声明为static;
懒汉式单例类
(第一次用到实例时才初始化,所以懒)
懒汉模式一:
(简单粗暴版,性能较差, 每个获取对象的线程都需要加锁)
public class LazySingleton {
//类加载时不实例化
private static LazySingleton instance = null;
//私有构造函数
private LazySingleton(){}
//synchronized同步
public synchronized static LazySingleton getInstance(){
if (instance==null) {
instance = new LazySingleton();
}
return instance;
}
}
懒汉模式二
(性能较好版:只有实例为null时才加锁)
public class Singleton {
//提供静态引用
private static Singleton singleton = null;
//私有化构造函数
private Singleton() {}
//提供静态方法获取引用
public static Singleton getSingleton() {
if (singleton == null) {
synchronized (Singleton.class) {
if (singleton == null) {
singleton = new Singleton();
}
}
}
return singleton;
}
}
注意点:
- 必须显式声明private构造函数,避免该单例类被其他类直接new对象、也避免源码编译时被添加默认构造函数。
2.懒汉式单例类中,getInstance函数要使用static和synchronized修饰,保证多线程环境下安全执行。
(线程同步概念:就是保证在多个线程环境下,同一个时刻只有一个线程操作该资源,比如使用synchronized修饰一个方法,那么系统运行时jvm会保证在一个时刻最多只有一个线程操作该方法)
单例模式使用场景:
- 要求系统产生唯一序号的环境
- 在整个项目中需要一个共享访问点或共享数据,比如web中的计数器。
3.创建一个对象需要消耗资源过多,如访问IO和数据库等资源。 - 需要定义大量静态常量和静态方法的环境。