package org.hyron.dp;
import java.util.Objects;
/**
* 懒汉式,线程不安全.
* 当有多个线程并行调用 getInstance() 的时候,
* 就会创建多个实例。也就是说在多线程下不能正常工作.
*
* @author lichunguang
*/
/*
public class Singleton {
private static Singleton instance;
// 构造函数私有化,拒绝使用new来创建对象
private Singleton() {
}
// 对外接口
public static Singleton getInstance() {
if (Objects.isNull(instance)) {
instance = new Singleton();
}
return instance;
}
}
*/
/**
* 懒汉式,线程安全.
* 效率低,每次调用都要判断锁。
* @author lichunguang
*/
/*
public class Singleton {
private static Singleton instance;
// 构造函数私有化,拒绝使用new来创建对象
private Singleton() {
}
// 对外接口
public static synchronized Singleton getInstance() { //同步方法
if (Objects.isNull(instance)) {
instance = new Singleton();
}
return instance;
}
}
*/
/**
* 双重检验锁.
* 这段代码看起来很完美,很可惜,它是有问题。
* 主要在于instance = new Singleton()这句,这并非是一
* 个原子操作,事实上在 JVM 中这句话大概做了下面 3 件事情。
*
* 给 instance 分配内存
* 调用 Singleton 的构造函数来初始化成员变量
* 将instance对象指向分配的内存空间(执行完这步 instance 就为非 null 了)
* 但是在 JVM 的即时编译器中存在指令重排序的优化。
* 也就是说上面的第二步和第三步的顺序是不能保证的,
* 最终的执行顺序可能是 1-2-3 也可能是 1-3-2。如果是后者,
* 则在 3 执行完毕、2 未执行之前,被线程二抢占了,
* 这时 instance 已经是非 null 了(但却没有初始化),
* 所以线程二会直接返回 instance,然后使用,
* 然后顺理成章地报错。
* @author lichunguang
*/
/*
public class Singleton {
private static Singleton instance;
// 构造函数私有化,拒绝使用new来创建对象
private Singleton() {
}
// 对外接口
public static Singleton getInstance() {
if(Objects.isNull(instance)){
synchronized(Singleton.class){//同步语句块
if (Objects.isNull(instance)) {
instance = new Singleton();
}
}
}
return instance;
}
}
*/
/**
* 饿汉式,静态域.
* 缺点:不是一种懒加载模式(lazy initialization).
* @author lichunguang
*/
/*
public class Singleton {
private static final Singleton instance=new Singleton();
// 构造函数私有化,拒绝使用new来创建对象
private Singleton() {
}
// 对外接口
public static Singleton getInstance() {
return instance;
}
}
*/
/**
* 静态内部类 static nested class.
* 推荐
* @author lichunguang
*/
/*
public class Singleton {
private static class InstanceHolder{
private static final Singleton instance=new Singleton();
}
// 构造函数私有化,拒绝使用new来创建对象
private Singleton() {
}
// 对外接口
public static final Singleton getInstance() {
return InstanceHolder.instance;
}
}
*/
/**
* Enum.
* 可以通过Singleton.INSTANCE来访问实例
*
* @author lichunguang
*/
public enum Singleton{
INSTANCE;
}
单例模式
最后编辑于 :
©著作权归作者所有,转载或内容合作请联系作者
- 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
- 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
- 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
推荐阅读更多精彩内容
- 当我们使用单例模式,获取单例的时候经常见到下面这种写法: 为什么会这么写呢,原因是为了避免多线程并发的时候创建多余...
- 单例模式4:多线程二(双重锁定)这种双重锁定考虑了线程安全,是正规写法 游戏常用设计模式之单例设计模式的写法大概常...
- 单例模式的作用可以保证在程序运行过程,一个类只有一个实例,而且该实例易于供外界访问从而方便地控制了实例个数,并节约...
- 对象的 2 大特征 属性 方法 面向对象的特点 封装:低耦合高内聚,即:同样的功能,只需要写一次,把它封装起来,以...