什么是单例?
单例(Singleton)对象的类必须保证只有一个实例存在——这是维基百科上对单例的定义。
对单例的实现可以分为两大类——懒汉式和饿汉式。
懒汉式和饿汉式的区别:
懒汉式:指全局的单例实例在第一次被使用是构建。
饿汉式:指全局的单例实例再类装载是构建。
懒汉式单例:
上述单例有关的知识点:
把构造器改称为私有的,这样能够防止被外部的类调用。
加上synchronized关键字后,getSingle方法会锁定。如果有两个线程,(T1,T2)同时执行这个方法时,会有其中一个线程T1获得同步锁,得以继续执行,而另一个线程T2则需要等待,当T1执行完毕getSingle之后(完成了Null判断,对象创建,或得返回值之后),T2线程才会执行。
有两次if(single== null)的判断,这个叫做双重检查【Double-Check】。
什么是原子操作:
原子操作(atomic)就是不可分割的操作,在计算机中,就是指不会因为线程调动而打断的操作。
什么是指重排:
简单来说,就是计算机为了提高执行效率,会做的一些优化,在不影响最终的情况下,可能会对一些语句的执行顺序做一些调整。
volatile关键字的一个作用是禁止指重排。
饿汉式单例:
对于饿汉式单例来,基本是完美的。
使用他的缺点就只是饿汉式单例本身的存在了 ——由于single的初始化是在类加载时进行的,而类的加载是由ClassLoader来做的,所以开发者对于他的初始化的时机会很难去把握:
- 可能由于初始化的太早,造成资源的浪费.
- 如果初始化本身依赖于一些其他数据,那么也就很难保证其他数据会在它初始化之前准备好。 当然,如果所需的单例占用的资源很少,并且也不依赖于其他数据,那么这种实现方式也是很好的。
什么时候是类装载时?
大致有这几个条件时,会触发一个类别加载:
1.new 一个对象时。
2.使用反射创建他的实例时。
3.子类被加载时,如果父类没被加载,就先加载父类。
4.jvm启动时执行的主类会首先被加载。
一些其他的实现方式:
1.Effective java1-静态内部类
2.Effective java-2 枚举:
public enum SingleD {
SINGLE_D;
public void getSingD(){
//do something
}
}
//使用
SingleD.SINGLE_D.getSingD();
这是一个枚举类型,连class都没用。
这种写法在功能上与共有域方法相近,但他更简洁,无偿的提供了序列化机制,绝对防止对此实例化,及时在面对复杂得序列化和反射攻击的时候。