单例模式介绍
A.单例模式定义:
所谓单例,就是整个程序有且仅有一个实例,该类负责创建自己的对象,同时姚确保只有一个对象被创建
好处就是一个类你只能创建一个实例对象可以节约内存开销。
一般常用语在工具类的实现。(之前讲的静态方法也可以用于工具类的实现)。
单例模式特点:
1.构造器私有
2.持有自己类型的属性
3.对外提供获取实例的静态方法
一般单例模式分为两大类:懒汉模式、饿汉模式。详细分其实有懒汉、饿汉、双检索、内部类、枚举实现等方式
懒汉模式:
package 单例;
/**
* 懒汉模式,延迟初始化,在调用发给发获取时才会实例对象
* 线程不安全的,严格意义上不是单例模式
* @author Irving
*
*/
public class Singleton {
//持有自己类型的属性
private static Singleton a;
//类构造器私有
private Singleton() {}
//对外提供获取实例的静态方法
public static Singleton getInstance() {
if (a==null) {
a=new Singleton();
}
return a;
}
}
饿汉模式:
线程安全,比较常用,但是容易产生垃圾。因为饿汉模式是一开始加载类的时候就初始化了实例
package 饿汉模式单例;
public class Singleton {
private static Singleton instance= new Singleton();
private Singleton() {}
public static Singleton getInstance() {
return instance;
}
}
懒汉模式和饿汉模式区别:
懒汉是非线程安全,饿汉是线程安全。
懒汉模式是一开始不实例对象,调用方法时才会实例化对象,饿汉模式一开始就实例化了对象,容易造成资源浪费。
综合两者各自优缺点,可以使用双检索模式,综合了懒汉和饿汉两者的优缺点整合而成
双检索模式:
双检索也称为双重校验锁,综合了懒汉和饿汉两者的优缺点整合而成。
package 双检索模式;
/**
*双检索模式结合懒汉和饿汉模式的优点:延迟加载而且线程安全
*
*/
public class Singleton {
private static volatile Singleton instance;
private Singleton() {}
public static Singleton getInstance() {
if (instance==null) {
synchronized (Singleton.class) {
if (instance==null) {
instance=new Singleton();
}
}
}
return instance;
}
}
双检索模式,进行了两次判断,第一次判断时为了避免不要的实例,第二次是为了进行线程同步,避免多线程问题。
由于 new Singleton()创建对象的时候jvm中可能会重新排序,在多线程下存在风险,使用volatile关键字可以当线程改变其值后通知其他线程改变并且不会被jvm重新排序,解决该问题。
E.内部类模式
package 内部类模式;
public class Singleton {
private Singleton() {}
public static Singleton getInstance() {
return Inner.instance;
}
public static class Inner{
//final修饰,不能改变
private static final Singleton instance=new Singleton();
}
}
只有第一次调用getInstance()方法时,虚拟机才加载inner内部类并初始化instance,只有一个线程可以获得instance对象的初始化,其他线程无法再进行初始化,保证对象的唯一性。【推荐使用此方式使用单例模式】
F.枚举实现单例
package 枚举;
/**
* 枚举:默认的枚举实例时线程安全的,并且再任何情况下都是单例的
* 枚举类默认隐藏了私有的构造方法
* @author Irving
*
*/
public enum Singleton {
INSTANCE;
public static Singleton getInstance() {
return Singleton.INSTANCE;
}
}
本文深入解析单例模式的定义、特点及其实现方式,包括懒汉模式、饿汉模式、双检索模式、内部类模式和枚举实现。探讨每种模式的优缺点,帮助读者理解并正确应用单例模式。
1237

被折叠的 条评论
为什么被折叠?



