前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >设计模式之单例模式

设计模式之单例模式

原创
作者头像
tanoak
发布2018-07-30 20:08:23
4570
发布2018-07-30 20:08:23
举报
文章被收录于专栏:java闲聊java闲聊

应用场景

  1. Windows的Task Manager(任务管理器)就是很典型的单例模式(这个很熟悉吧),想想看,是不是呢,你能打开两个windows task manager吗? 不信你自己试试看哦~
  2. windows的Recycle Bin(回收站)也是典型的单例应用。在整个系统运行过程中,回收站一直维护着仅有的一个实例。
  3. 操作系统的文件系统,也是大的单例模式实现的具体例子,一个操作系统只能有一个文件系统。

在Spring管理的Bean中,默认就是单例存在的,让我们来深入了解单例模式

举栗子,在小说神雕侠侣中只会有一个小龙女不然那就成西游记,一根毫毛一个分身

饿汉模式

package com.tanoak.create.single;
/**
 * @author tanoak@qq.com
 * @date 2018/7/18 15:29
 * @Desc 小龙女  单例 =>饿汉模式  不会存在多线程并发问题 但是占用内存
 */
public class DragonGirl1 {

	private String name ;
	private String sex ;
	private String desc ;
	private DragonGirl1(){}
	private static DragonGirl1 girl = new DragonGirl1();
	public static DragonGirl1 getInstance() {
		return girl;
	}
}

懒汉模式

/**
 * @author tanoak@qq.com
 * @date 2018/7/18 15:29
 * @Desc 小龙女  单例 =>懒汉模式
 */
public class DragonGirl2 {

	private String name ;
	private String sex ;
	private String desc ;
	private DragonGirl2(){}
	private static DragonGirl2 girl  = null;

	public static DragonGirl2 getInstance(){
		if (null == girl) {
			girl = new DragonGirl2();
		}
		return girl;
	}

}

会存在并发问题,因为调用newInstance()方法时没有加锁,导致会并发执行,如图:

2a.png
2a.png

双重检查机制

/**
 * @author tanoak@qq.com
 * @date 2018/7/18 15:29
 * @Desc 小龙女  单例 =>双重检查机制
 */
public class DragonGirl3 {
	private String name ;
	private String sex ;
	private String desc ;
	private DragonGirl3(){}
	private static volatile  DragonGirl3 girl  = null;

	public  static DragonGirl3 getInstance(){
		if (null == girl) {
			synchronized (DragonGirl3.class){
				if(null ==girl){
					girl = new DragonGirl3(); //1
				}
			}
		}
		return girl;
	}
}

这种方式是最常用的,我也比较喜欢用这种方式,在这里注意volatile这个关键字,

作用:禁止指令重排序(不对volatile做详细介绍),详细了解volatile可以参考这篇博文volatile详解

为什么要禁止指令重排序?

这是因为在jdk1.5之后,JAVA是无序写入,可能会造成顺序的颠倒,即内存分配、返回对象引用、初始化的顺序,这种情况下对应到//1就是singleton已经不是null,而是指向了堆上的一个对象,但是该对象却还没有完成初始化动作。当后续的线程发现singleton不是null而直接使用的时候,就会出现问题。

测试

/**
 * @author tanoak@qq.com
 * @date 2018/7/18 15:29
 * @Desc 单例模式
 */
public class Main {
	public static void main(String[] args) throws Exception {
		for (int i = 0; i < 50; i++) {
			createThread();
		}
	}
	private static void createThread() throws Exception{
		new Thread(() -> {
			System.out.println(Thread.currentThread().getName()+" :=> "+DragonGirl1.getInstance());
//			System.out.println(Thread.currentThread().getName()+" :=> "+DragonGirl2.getInstance());
//			System.out.println(Thread.currentThread().getName()+" :=> "+DragonGirl3.getInstance());
			try {
				Thread.sleep(6000);
			} catch (InterruptedException e) {
				e.printStackTrace();
			}
		}).start();
	}
}

在Java中还可以用枚举类型来实现单例,但是日常的开发中使用的比较少,主要还是习惯原因,想了解的可以参考枚举单例这篇博文

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

原创声明:本文系作者授权腾讯云开发者社区发表,未经许可,不得转载。

如有侵权,请联系 cloudcommunity@tencent.com 删除。

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 应用场景
  • 饿汉模式
  • 懒汉模式
  • 双重检查机制
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档