主要讲解单例模式和常用的几种实现方式,基于java。
我理解单例应该是所有模式中,最简单,也是使用最多的一种,我就简单总结一下,首先了解一下单例模式的定义。
确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例。
由单例的定义,可以分析出,实现一个单例,有以下几个要点:
单例主要有两种种实现方式,懒汉模式和饿汉模式。
在类加载时,不创建实例,因此类加载速度快,但运行时获取对象的速度慢,代码如下:
public class penguin {
private static volatile penguin m_penguin = null;
// 避免通过new初始化对象
private void penguin() {}
public void beating() {
System.out.println("打豆豆");
};
public static penguin getInstance() {
if (null == m_penguin) {
synchronized(penguin.class) {
if (null == m_penguin) {
m_penguin = new penguin();
}
}
}
return m_penguin;
}
}
懒汉模式实现要点
下面模拟一个简单的单例并发测试,可以使用CountDownLatch,使用await()等待锁释放,使用countDown()释放锁从而达到并发的效果,可以见下面的代码:
public static void main(String args[]) {
final CountDownLatch latch = new CountDownLatch(1);
int threadCount = 20;
for (int i = 0; i < threadCount; i++) {
new Thread() {
@Override
public void run() {
try {
latch.await();
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println(penguin.getInstance().hashCode());
}
}.start();
}
latch.countDown();
}
打印出来的hashCode完全一样,证明单例模式生效,输出如下:
1449937592
1449937592
1449937592
1449937592
1449937592
1449937592
1449937592
1449937592
1449937592
1449937592
在类加载时就完成了初始化,所以类加载较慢,但获取对象的速度快,代码如下:
public class penguin {
private static penguin m_penguin = new penguin();
private void penguin() {}
public static penguin getInstance() {
return m_penguin;
}
}
两种实现模式各有优缺点,综合来说,个人比较偏向于懒汉模式。
单例模式只允许创建一个对象,因此节省内存,加快对象访问速度,因此对象需要被公用的场合适合使用,如多个模块使用同一个数据源连接对象等等。如:
优点:
缺点:
大家有没有注意到,单例模式中用到了关键字volatile,在PHP和Go中没有类似的关键字,但是JAVA必须加,当初还有疑问,我们先看一下volatile的作用:
volatile是Java提供的一种轻量级的同步机制。Java 语言包含两种内在的同步机制:同步块(或方法)和 volatile 变量,相比于synchronized(synchronized通常称为重量级锁),volatile更轻量级,因为它不会引起线程上下文的切换和调度。但是volatile 变量的同步性较差(有时它更简单并且开销更低),而且其使用也更容易出错。
我直接总结一下volatile的作用:
再回顾一下单例模式代码:
public class penguin {
private static volatile penguin m_penguin = null;
// 避免通过new初始化对象
private void penguin() {}
public void beating() {
System.out.println("打豆豆");
};
public static penguin getInstance() { //1
if (null == m_penguin) { //2
synchronized(penguin.class) { //3
if (null == m_penguin) { //4
m_penguin = new penguin(); //5
}
}
}
return m_penguin; //6
}
}
在并发情况下,如果没有volatile关键字,在第5行会出现问题。instance = new TestInstance();可以分解为3行伪代码:
a. memory = allocate() //分配内存
b. ctorInstanc(memory) //初始化对象
c. instance = memory //设置instance指向刚分配的地址
上面的代码在编译运行时,可能会出现重排序从a-b-c排序为a-c-b。在多线程的情况下会出现以下问题。当线程A在执行第5行代码时,B线程进来执行到第2行代码。假设此时A执行的过程中发生了指令重排序,即先执行了a和c,没有执行b。那么由于A线程执行了c导致instance指向了一段地址,所以B线程判断instance不为null,会直接跳到第6行并返回一个未初始化的对象。
关于volatile关键字,后面我会单独拿一篇文章进行讲解~~
单例模式其实还有其它的实现方式,但是主要就用到“懒汉模式”,其它的实现方式,大家感兴趣的话,可以到网上再查一下。其实本来不想写单例模式,因为感觉太简单了,然后这篇文章,应该也是我所有文章中,写的最快的一篇(五一出去旅游,晚上在酒店很快写完了,老婆对我出去玩,回来还抱个电脑很有意见),那就通过这篇文章,简单记录一下吧。
欢迎大家多多点赞,更多文章,请关注微信公众号“楼仔进阶之路”,点关注,不迷路~~