目录
单例模式在java编程的过程中经常被使用到,比如日常开发中,数据库的连接池不会反复创建,及在spring框架中创建的bean默认都是单例的。
一、单例模式概念:
如果一个类只允许创建一个对象(实例),保证在一个 JVM 中只存在单一实例,那这个类就是一个单例类,这种设计模式就叫单例模式。
作用:主要解决一个全局的使用类被频繁地创建与销毁问题,,从而提升代码的整体性能;
种类:饿汉式、懒汉式、双重检测、静态内部类、枚举,可以根据需要去选择要使用的方式。
二、单例模式的组成套路(3个条件)
1.单例类只能有一个实例
2.单例类必须自己创建自己的唯一实例;
3.单例类必须给其它对象提供获取该实例的方法
三:几种类型示例
1.懒汉式:
特点:
- lazy初始化
- getInstance方法无synchronized线程不安全
- getInstance有synchronized线程安全,但是效率很低;
- 容易实现
public class Singleton {
//创建 SingleObject 的一个对象
private static Singleton instance;
//让构造函数为 private,这样该类就不会被实例化
private Singleton (){}
//获取唯一可用的对象
public static synchronized Singleton getInstance() {
if (instance == null) {
instance = new Singleton();
}
return instance;
}
}
2.饿汉式:
- 类加载时候就初始化
- 没有加锁,执行效率提高
- 天生就是线程安全的,JVM初始化类的过程中,保证了不会初始化多个static对象
public class Singleton {
private static Singleton instance = new Singleton();
private Singleton (){}
public static Singleton getInstance() {
return instance;
}
}
3.双重检测(DCL:double checked locking)
- 懒加载
- 线程安全
- 多线程情况下较高性能
public class Singleton {
private volatile static Singleton singleton;
private Singleton (){}
public static Singleton getSingleton() {
if (singleton == null) {
synchronized (Singleton.class) {
if (singleton == null) {
singleton = new Singleton();
}
}
}
return singleton;
}
}
4.静态内部类
- 懒加载
- 线程安全
- 多线程情况下较高性能
- 比双重检测更合理
public class Singleton {
// 静态内部类
private static class SingletonHolder {
private static final Singleton INSTANCE = new Singleton();
}
private Singleton (){}
public static final Singleton getInstance() {
return SingletonHolder.INSTANCE;
}
}
5.枚举方式
- 非懒加载
- 线程安全
- 自动支持序列机制,防止多次实例化
- 简洁清晰,被称为单例的最佳方法
public enum Singleton {
INSTANCE;
public void whateverMethod() {
// 具体的业务逻辑,do something ......
}
}
// 方法调用示例
public class Main {
public static void main(String[] args) {
Singleton.INSTANCE.whateverMethod();
}
}
四、本节总结
虽然单例模式是一个很平常的模式,但在各种方式的实现上却需要java的基本功;
看了那么多的介绍,相信有的人选择困难症就犯了,其实大可不必,这里比较推荐第4种静态内部类的单例模式;
或者枚举单例模式,它是《Effective Java》作者Joshua J.Bloch推荐的模式,Joshua J.Bloch是美丽国著名的程序员,也是Google公司的首席架构师,还是值得我们相信的。