单例模式总结


昨天看了下单例模式的一些内容,就想做下总结。巩固一下自己所学的知识。在写之前先说一下什么是设计模式?简单的理解就是前人的经验总结。通过设计模式可以让我们的代码复用性更高,可维护性更高,让你的代码写的更优雅。

单例模式的目的:

希望对象只创建一个实例,并且提供一个全局的访问点。


运用的场合:

有些对象只需要一个就足够了,如线程池、缓存、配置文件、工具类、日志对象等。如果创造出多个实例,就会导致出现一些问题,比如:占用过多资源,不一致的结果等。


作用:

保证整个应用程序中某个实例有且只有一个。


类型:

常用的单例模式主要有两种:饿汉模式、懒汉模式。

饿汉模式(线程安全)

public class Singleton {
	//1.将构造方法私有化,不允许外部直接创建对象
	private Singleton(){		
	}
	
	//2.创建类的唯一实例,使用private static修饰
	private static Singleton instance=new Singleton();
	
	//3.提供一个用于获取实例的方法,使用public static修饰
	public static Singleton getInstance(){
		return instance;
	}
}
代码很简单,这就是饿汉模式。在加载类的时候就会创建类的实例。如果单例对象初始化非常快,而且占用内存非常小的时候这种方式是比较合适的,可以直接在应用启动时加载并初始化。

这到底是不是真的只有一个实例呢?我们可以简单的写一个Test类测试一下。代码如下

public class Test {
	public static void main(String[] args) {
		//饿汉模式
		Singleton s1=Singleton.getInstance();
		Singleton s2=Singleton.getInstance();
		if(s1==s2){
			System.out.println("s1和s2是同一个实例");
		}else{
			System.out.println("s1和s2不是同一个实例");
		}
}

输出的结果是:s1和s2是同一个实例。所以是真的只有一个实例。


懒汉模式(线程不安全)

public class Singleton {
	//1.将构造方式私有化,不允许外边直接创建对象
	private Singleton(){
	}
	
	//2.声明类的唯一实例,使用private static修饰
	private static Singleton instance;
	
	//3.提供一个用于获取实例的方法,使用public static修饰
	public static Singleton getInstance(){
		if(instance==null){
			instance=new Singleton();
		}
		return instance;
	}
}

这就是懒汉模式。加载类的时候只是声明类的实例,需要使用的时候再实例化。如果单例用到次数不是很多,但是这个单例提供的功能又非常复杂,而且加载和初始化要消耗大量的资源,这个时候使用懒汉式就是非常不错的选择。


区别

饿汉模式的特点是加载类时比较慢,但运行时获取对象的速度比较快,线程安全。懒汉模式的特点是加载类时比较快,但运行时获取对象的速度比较慢,线程不安全。


懒汉模式改进方法:

上面介绍的懒汉模式是线性不安全的。我们来说一下什么情况下这种写法会有问题。在运行过程中可能存在这么一种情况:有多个线程去调用getInstance方法来获取Singleton的实例,那么就有可能发生这样一种情况当第一个线程在执行if(instance==null)这个语句时,此时instance是为null的进入语句。在还没有执行instance=new Singleton()时(此时instance是为null的)第二个线程也进入if(instance==null)这个语句,因为之前进入这个语句的线程中还没有执行instance=new Singleton(),所以它会执行instance=new Singleton()来实例化Singleton对象,因为第二个线程也进入了if语句所以它也会实例化Singleton对象。这样就导致了实例化了两个Singleton对象。所以单例模式的懒汉式是存在线程安全问题的,既然它存在问题,那么可能有解决这个问题的方法,那么究竟怎么解决呢?对这种问题可能很多人会想到加锁于是出现了下面这种写法。

加锁的懒汉模式(线性安全,效率低,不推荐)

public class Singleton {
	//1.将构造方式私有化,不允许外边直接创建对象
	private Singleton(){
	}
	
	//2.声明类的唯一实例,使用private static修饰
	private static Singleton instance;
	
	//3.提供一个用于获取实例的方法,使用public static synchronized修饰
	public static synchronized Singleton getInstance(){
		if(instance==null){
			instance=new Singleton();
		}
		return instance;
	}
}
只是多了一个synchronized修饰,但是这种懒汉模式是线性安全的,只是效率很低,不推荐使用。每个线程在想获得类的实例时候,执行getInstance()方法都要进行同步。而其实这个方法只执行一次实例化代码就够了,后面的想获得该类实例,直接return就行了。方法进行同步效率太低要改进。
于是,我们可能会想到下面这种方式。

进一步改写的懒汉模式(线性不安全,不推荐)

public class Singleton {
	// 1.将构造方式私有化,不允许外边直接创建对象
	private Singleton() {
	}

	// 2.声明类的唯一实例,使用private static修饰
	private static Singleton instance;

	// 3.提供一个用于获取实例的方法,使用public static synchronized修饰
	public static Singleton getInstance() {
		if (instance == null) {
			synchronized (Singleton.class) {
				instance = new Singleton();
			}
		}
		return instance;
	}
}

可是这种方法也是线性不安全的。当一个线程还没有实例化Singleton时另一个线程执行到if(instance==null)这个判断语句时就会进入if语句,虽然加了锁,但是等到第一个线程执行完instance=new Singleton()跳出这个锁时,另一个进入if语句的线程同样会实例化另外一个Singleton对象,线程不安全的原理跟上面的类似。

最后,我们想到了双重校验锁。这种方法是推荐使用的懒汉模式!

懒汉模式双重校验锁(线性安全,效率较高,推荐使用)

public class Singleton {
	// 1.将构造方式私有化,不允许外边直接创建对象
	private Singleton() {
	}

	// 2.声明类的唯一实例,使用private static修饰
	private static Singleton instance;

	// 3.提供一个用于获取实例的方法,使用public static synchronized修饰
	public static Singleton getInstance() {
		if (instance == null) {
			synchronized (Singleton.class) {
				if (instance == null) {
					instance = new Singleton();
				}
			}
		}
		return instance;
	}
}



其实还有两种单例的写法:静态内部类和枚举。有兴趣的自己去查看下资料吧。


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值