五种单例模式

今天算是完完整整把老高的单例模式看了,差不多也懂了,个别不懂的再补吧,下面介绍一下5种单例模式:

  饿汉式实现:线程安全,调用效率高。但是,不能延时加载。

  懒汉式实现:线程安全,调用效率不高。但是,可以延时加载。

  双重检测锁式:由于JVM底层内部模型原因,偶尔会出现问题,不建议使用。

  静态内部类式:线程安全,调用效率高,可以延时加载。

  枚举式:线程安全,调用效率高。但是,不能延时加载(避免反射和反序列化的漏洞)。

以下先介绍一下5种单例模式:

  饿汉式实现:

/**
 *单例模式:饿汉模式
 *线程安全,调用效率高。但是,不能延时加载
 *
 * @author cookie
 *
 */
public class SingletonDemo01 {
	//类初始化时,立刻加载这个对象(没有延时加载的优势)。线程是天然安全
	private static SingletonDemo01 instance = new SingletonDemo01();
	private SingletonDemo01(){}
	//方法没有同步,线程安全,调用效率高。但是,不能延时加载
	public static SingletonDemo01 getInstance(){
		return instance;
	}
}

  懒汉式实现:

/**
 * 单例模式:懒汉模式
 * 线程安全,调用效率不高。但是,可以延时加载
 * 真正用的时候才加载,资源利用率高了。
 * @author cookie
 *
 */
public class SingletonDemo02 {
	//类加载时,不初始化对象(延时加载:资源利用率高)
	private static SingletonDemo02 instance;
	private SingletonDemo02(){}

	//synchronized 防止并发量高的时候,出现多个对象
	//方法同步,调用效率低,
	public static synchronized SingletonDemo02 getInstance(){
		if(instance==null){//真正用的时候才加载
			instance = new SingletonDemo02();
		}
		return instance;
	}
}

  双重检测锁式

/**
 * 单例模式:双重检验锁式 结合懒汉式和饿汉式的优点,但由于JVM底层内部模型原因,偶尔会出现问题,不建议使用
 *
 * @author cookie
 *
 */
public class SingletonDemo03 {
	private volatile static SingletonDemo03 instance = null;

	private SingletonDemo03() {
	}

	public static SingletonDemo03 getInstance() {
		if (instance == null) {
			synchronized (SingletonDemo03.class) {// 1
				if (instance == null) {// 2
					instance = new SingletonDemo03();// 3
				}
			}
		}
		return instance;
	}
}

  静态内部类式

/**
 * 单例模式:静态内部类实现单例模式。 优点:线程安全,调用效率高,实现延时加载。
 *
 * @author cookie
 *
 */
public class SingletonDemo04 {
	// 类加载时静态内部类不会加载,只有调用getInstance方法时,才会加载(实现延时加载)
	private static class SingletonClassInstance {
		private final static SingletonDemo04 instance = new SingletonDemo04();
	}

	private SingletonDemo04() {
	}

	// 线程安全,方法不同步,调用效率提高
	public static SingletonDemo04 getInstance() {
		return SingletonClassInstance.instance;
	}
}

  枚举式:

/**
 * 单例模式:枚举类型实现单例模式 线程安全,调用效率高。但是,不能延时加载 避免反射和反序列化的漏洞
 *
 * @author cookie
 *
 */
public enum SingletonDemo05 {
	INSTANCE;// 这个枚举元素,本身就是单例模式
	// 添加自己需要的操作
	public void singletonOperation() {

	}

}

  下面是测试5种单例模式的效率:

/**
 *
 * 描述:多线程环境下五种单例模式的效率
 * @author cookie
 */
public class Client2 {

	/**
	 * @param args
	 * @throws InterruptedException
	 */
	public static void main(String[] args) throws InterruptedException {
		long start = System.currentTimeMillis();
		int threadNum = 10;
		final CountDownLatch countDownLatch = new CountDownLatch(threadNum);
		//10个线程
		for (int i = 0; i < threadNum; i++) {

			new Thread(new Runnable() {
				@Override
				public void run() {
					for (int i = 0; i < 100000; i++) {
						SingletonDemo05 s1 = SingletonDemo05.INSTANCE;
					}
					countDownLatch.countDown();//计数器减一
				}
			}).start();

		}
		countDownLatch.await();//main线程阻塞,等待计数器为0,即threadNum = 0;
		long end = System.currentTimeMillis();
		System.out.println("总耗时:"+(end-start));
	}

}

  实现结果如下:

单例模式 总耗时
饿汉式 13
懒汉式 547
双重检测锁式 20
静态内部类式 17
枚举式 13

 选用时,

  单例对象,占用资源少,无需延时加载,枚举式 优于 饿汉式

  单例对象,占用资源多,需要延时加载,静态内部类式 优于 懒汉式

时间: 2024-12-29 07:13:23

五种单例模式的相关文章

Java中的五种单例模式实现方法

[代码] Java中的五种单例模式实现方法 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 package s

五种单例模式实现

核心作用: 保证一个类只有一个实例,并向外提供一个访问该实例的访问点. 常见场景: 数据库连接池的设计一般也是单例模式 在Servlet编程中,每个Servlet也是单例模式 在Spring中,默认创建的bean也是单例模式 ...... 优点: 1.由于每个类只创建一个实例,大大减少了内存的开销. 2.单例模式提供全局访问点,可以实现共享资源访问. 常见的五种单例模式实现方法: 饿汉式(线程安全,效率高,不能延迟加载) 懒汉式(线程安全,效率不高,可以延迟加载) DCL懒汉式(线程安全,效率还

五种单例模式----来自脚本之家

本文为大家分享了Python创建单例模式的5种常用方法,供大家参考,具体内容如下 所谓单例,是指一个类的实例从始至终只能被创建一次. 方法1: 如果想使得某个类从始至终最多只有一个实例,使用__new__方法会很简单.Python中类是通过__new__来创建实例的: ? 1 2 3 4 5 6 7 8 9 10 11 12 13 class Singleton(object):   def __new__(cls,*args,**kwargs):     if not hasattr(cls,

快速理解Java中的五种单例模式

解法一:只适合单线程环境(不好) package test; /** * @author xiaoping * */ public class Singleton { private static Singleton instance=null; private Singleton(){ } public static Singleton getInstance(){ if(instance==null){ instance=new Singleton(); } return instance;

常见的五种单例模式实现方式

--主要: 饿汉式(线程安全,调用效率高,但是不能延时加载) 懒汉式(线程安全,调用效率低,但是可以延时加载) --其他: 双重检测锁式(由于JVM底层内部模型原因,偶尔会出现问题,不建议使用) 静态内部类式(线程安全,调用效率高,可以延时加载) 枚举式(线程安全,调用效率高,不能延时加载) -- 如何选用? 单例对象   占用   资源 少,不需要   延时加载:  枚举式   好于  饿汉式 单例对象   占用   资源 大, 需要   延时加载:   静态内部类式   好于   懒汉式 引

C# 单例模式的五种写法

C# 单例模式的五种写法及优劣分析,见下文: [单例模式及常见写法](http://blog.csdn.net/jiankunking/article/details/50867050)

单例模式的五种写法

首先来明确一个问题,那就是在某些情况下,有些对象,我们只需要一个就可以了,比如,一台计算机上可以连好几个打印机,但是这个计算机上的打印程序只能有一个,这里就可以通过单例模式来避免两个打印作业同时输出到打印机中,即在整个的打印过程中我只有一个打印程序的实例. 简单说来,单例模式(也叫单件模式)的作用就是保证在整个应用程序的生命周期中, 任何一个时刻,单例类的实例都只存在一个(当然也可以不存在). package singleton; /** * @author lei * 单例模式的五种写法: *

五种方法实现Java的Singleton单例模式

面试的时候经常会问到Java的单例模式,这道题能很好的考察候选人对知识点的理解程度.单例模式要求在系统运行时,只存在唯一的一个实例对象. 下面我们来详细剖析一下其中的关键知识点,并介绍五种实现方法,以及它们的优缺点. 一.最简单的方法是在类加载的时候初始化这个单独的实例. 首先,定义单例类(没有特别的,就是起个名字): 1 public class Singleton{ 其次,需要定义类变量将单例对象保存下来: 1 private static Singleton instance = new

php 常用五种模式

/* 设计模式之单例模式 $_instance 必须声明为静态的私有变量 构造函数必须声明为私有,防止外部程序 new 类从而失去单例模式的意义 getInstance() 方法必须设置为公有的,必须调用此方法以返回实例的一个引用 :: 操作符只能访问静态变量和函数 new 对象会消耗内存 使用场景:最常用的地方是数据库连接 使用单例模式生成一个对象后,该对象可以被其它众多对象所使用 [单例模式适用场景] 1.当类只能有一个实例而且客户可以从一个众所周知的访问点访问它时 2.当这个唯一实例应该是