package com.exam.test;
/**
* 懒汉式单例
* Created by xiangyang.laixiang on 2016/8/23.
*/
public class SingleInstance {
private static SingleInstance singleInstance;
/**
* 懒汉式单例,非线程安全
* @return
*/
public static SingleInstance getSingleInstance(){
if(singleInstance == null)
{
singleInstance = new SingleInstance();
}
return singleInstance;
}
/**
* 懒汉式单例,线程安全
* 这是实现线程安全最简单的方式,但这里会导致一个性能问题,因为我们把整个getInstance同步起来了,
* 这就会导致每时每刻都只能有一个线程调用getInstance方法,而同步只发生在第一次声明的时候
* 这就引出了双重检验锁的概念
* @return
*/
public synchronized static SingleInstance getSingleInstance2(){
if(singleInstance == null)
{
singleInstance = new SingleInstance();
}
return singleInstance;
}
}
/**
* 双重检验锁
*/
class DoubleCheckSingleInstance{
//使用volatile关键字修饰
//private static DoubleCheckSingleInstance singleInstance;
private static DoubleCheckSingleInstance singleInstance;
public static DoubleCheckSingleInstance getSingleInstance()
{
if(singleInstance == null)
{
synchronized (DoubleCheckSingleInstance.class)
{
if(singleInstance == null)
{
singleInstance = new DoubleCheckSingleInstance();
}
}
}
return singleInstance;
}
/**
* 上述这段代码看起来很完美,但是中间存在着一个问题那就是 new DoubleCheckSingleInstance()
* 这个操作不是原子操作,大致有三步构成。
* 1. 给instance分配内存
* 2. 调用DoubleCheckSingleInstance构造函数初始化成员变量
* 3. 将instance指向内存
* 问题就出在这一部分,jvm即时编译器中存在着指令重排序的优化。如果1,2,3的执行步骤不会存在问题
* 倘若1,3,2
* 那么当线程二执行完第三步以后,线程三抢占cpu资源,则进行检查,发现instance不为空,则返回,但此时
* 成员变量尚未初始化,则会发生调用错误。
* 解决方案是使用volatile来修饰singleInstance来强制每次都从内存映像中读取数据,其实volitile也可以起到
* 禁止重排序的功能,在java1.5以后使用比较安全。(1.5之前的内存模型是存在问题的)
*/
}
/**
* effective java中推荐的内部类实现单例的写法 。
*/
class SingtonInstance{
private SingtonInstance()
{
System.out.println("constructor");
}
private static class SingtonHolder{
private static SingtonInstance instance = new SingtonInstance();
}
public static void beforeInvoke()
{
System.out.println("before invoke");
}
public static SingtonInstance getInstance()
{
/**
* 此处最开始理解有点绕,但是经过我对象的一个提醒恍然明白,对于内部类的所有对象对于其
* 外部类来说都是可见的,简直是霸气,否则private的可见范围是不允许直接访问的。
*/
return SingtonHolder.instance;
}
}
public class Test {
public static void main(String[] args) {
SingtonInstance.beforeInvoke();
SingtonInstance.getInstance().beforeInvoke();
}
}
单例模式(懒汉模式)
最后编辑于 :
©著作权归作者所有,转载或内容合作请联系作者
- 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
- 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
- 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
推荐阅读更多精彩内容
- 当我们使用单例模式,获取单例的时候经常见到下面这种写法: 为什么会这么写呢,原因是为了避免多线程并发的时候创建多余...
- 单例模式4:多线程二(双重锁定)这种双重锁定考虑了线程安全,是正规写法 游戏常用设计模式之单例设计模式的写法大概常...