JAVA23种设计模式——单例模式

当初刚开始学习java的时候,对于单例模式只是知道两种,饿汉模式,懒汉模式。
后来慢慢接触才发现还有很多种实现方法,越发觉得java的设计模式体现着代码的美感,这里归纳总结一下实现单例模式的几种形式。

首先有几个要点:

  • 单例类只能有一个实例。
  • 单例类必须自己创建自己的唯一实例。
  • 单例类必须给所有其他对象提供这一实例。
  1. 饿汉模式(线程安全,但是容易产生垃圾对象,因为在类装载时就实例化,但不知是否会用到)
/*
 * 饿汉模式(未使用时就先实例化)
 * 线程安全
 */
public class Singleton {
	private Singleton(){}//private构造方法,保证不会在其它类里产生对象,实现单例的要点
	private static final Singleton singleton = new Singleton();
	public static Singleton getInstance(){
		return singleton;
	}
	public void sayHello(){
		System.out.println("hello");
	}
}
//调用
public class Test {
	public static void main(String[] args) {
		//获取唯一实例
		Singleton singleton = Singleton.getInstance();
		//doSomething
		singleton.sayHello();
	}
}
  1. 懒汉模式①(线程不安全)
    第一次调用时才初始化,当判断实例为空时则创建,但是在多线程的情况下, 有可能出现两个请求都是第一次且同时调用getInstance()方法,导致都是判断为空,创建多个实例。
/*
 * 懒汉模式(什么时候使用什么时候再实例化)
 * 线程不安全
 */
public class Singleton {
	private Singleton(){}//private构造方法,保证不会在其它类里产生对象
	private static Singleton singleton;
	public static Singleton getInstance(){
		if(singleton == null) singleton = new Singleton();
		return singleton;
	}
	public void sayHello(){
		System.out.println("hello");
	}
}
//调用
public class Test {
	public static void main(String[] args) {
		//获取唯一实例
		Singleton singleton = Singleton.getInstance();
		//doSomething
		singleton.sayHello();
	}
}
  1. 懒汉模式②(线程安全)
    加了synchronized 关键字,但是加锁会对性能有影响。
/*
 * 线程安全
 */
public class Singleton {
	private Singleton(){}//private构造方法,保证不会在其它类里产生对象
	private static Singleton singleton;
	public  static synchronized Singleton getInstance(){
		if(singleton == null) singleton = new Singleton();
		return singleton;
	}
	public void sayHello(){
		System.out.println("hello");
	}
}
//调用方法同上几种模式
  1. 双重校验锁机制(线程安全)
    先判断实例是否为空,为空再加锁,不为空直接返回实例,提高了效率。
    相比上一个懒汉模式②,不管是否为空都加锁,抢到锁的线程才能获取实例,对性能产生的影响。
public class Singleton {
	private Singleton(){}//private构造方法,保证不会在其它类里产生对象
	private static Singleton singleton;
	public static Singleton getInstance(){
		if(singleton == null){//先校验一次是否为空,此时还存在线程不安全因素
			synchronized(Singleton.class){//如果为空,则加锁之后进行下一步
				if(singleton == null) singleton = new Singleton();//再判断一次是否为空,防止多个线程重复创建实例,因为之前已经有了锁,所以只会创建一次,线程安全。
			}
		}
		return singleton;
	}
	public void sayHello(){
		System.out.println("hello");
	}
}
//调用方法同上
  1. 静态内部类(线程安全)
    相比于第一个例子饿汉模式(类加载时singleton 便实例化),但是使用静态内部类,只要没有被显式使用,则不会创建Singleton 的实例(new Singleton()),只有使用getInstance()方法时才会创建对象实例,则避免了浪费性能的情况
public class Singleton {
	private Singleton(){}//private构造方法,保证不会在其它类里产生对象
	private static class SingletonInternal{//静态内部类
	    private static final Singleton singleton = new Singleton();  
	}  
    public static final Singleton getInstance() {  
    	return SingletonInternal.singleton;//只有在此显式的调用时,Singleton 才会被实例化(new Singleton())
    }
    public void sayHello(){
		System.out.println("hello");
	}
}
//调用方法同上
  1. 枚举(线程安全)
    枚举的构造方法默认为private,枚举可以看成一个特殊的类,每一个枚举类型成员都可以看作是 Enum 类的实例,这些枚举值默认都被 public static final修饰,当只有一个枚举值时,可以看作Singleton有一个实例。它不仅能避免多线程同步问题,而且还自动支持序列化机制,防止反序列化重新创建新的对象,绝对防止多次实例化
public enum Singleton {  
    INSTANCE;
    public void sayHello() {  
    	System.out.println("hello");
    }  
}
//调用
public class Test {
	public static void main(String[] args) {
		Singleton.INSTANCE.sayHello();
	}
}

关于单例模式就这么多,其中枚举的方式实现单例比较简洁优雅,因为在jdk1.5之后枚举才加入枚举,说起单例模式仍是首先想到前面几种。
另外如果有兴趣一起进步学习,请搜索名称关注我的公众号:IT成长日记
或扫码关注:在这里插入图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值