单例模式----设计模式系列

简介:

这次主要写了四种类型的单例模式

  • 饿汉式
  • 懒汉式
  • 同步方法锁
  • 双重锁定
自己留着以后忘记时查看用的,所以注释比代码都多。项目结构很简单,一个Singleton类,一个测试类。直接上代码了。
package com.dyk.singleton;

public class Singleton {

	private static Singleton mSingleton;

	/**
	 * <b>饿汉式单例</b>
	 * <p>
	 * 静态初始化的方式是在自己被加载时就将自己实例化,所以被形象地称为饿汉式单例类
	 */
	private static Singleton mSingleton2 = new Singleton();

	/** <b>饿汉式单例</b> */
	public static Singleton getInstance5() {
		return mSingleton2;
	}

	/** 私有化构造方法,防止从外界直接创建对象。 */
	private Singleton() {

	}

	/**
	 * <b>懒汉式单例。</b>在多线程模型下可能产生多个实例。 例如:当两个线程同时运行到new Singleton()那一行,则会产生两个对象。
	 * @return instanceof Singleton
	 */
	public static Singleton getInstance() {
		if (mSingleton == null) {
			mSingleton = new Singleton();
		}
		return mSingleton;
	}

	/**
	 * 运用同步锁机制,锁住获取实例的方法。 优点:可以保证一个类中只有一个实例。
	 * 缺点:在多线程环境下,第一个线程进入getInstance2()方法后,其余线程都必须等待,直到第一个线程释放资源
	 * 假设有100个线程,那么第一百个线程需要等待前99个线程全部释放资源后才能进入getInstance2()方法
	 * 
	 * @return instanceof Singleton
	 */
	public static synchronized Singleton getInstance2() {
		if (mSingleton == null) {
			mSingleton = new Singleton();
		}
		return mSingleton;
	}

	/**
	 * 和getInstance2()很类似。
	 * 区别是getInstance2()锁住的是方法,在线程访问getInstance2()的同时,其他线程可以访问Singleton类的其他方法。
	 * 而getInstance3()锁住的是整个类,独占了整个类的访问。
	 * @return instanceof Singleton
	 */
	public static Singleton getInstance3() {
		synchronized (Singleton.class) {
			if (mSingleton == null) {
				mSingleton = new Singleton();
			}
		}
		return mSingleton;
	}

	/**
	 * <b>双重锁定(Double-Check-Locking)</b><br/>
	 * 在synchronized关键字内还要再进行一次判空是因为:<br/>
	 * 如果同时有两个线程调用GetInstace4()方法时,<br/>
	 * 它们都将可以通过第一重mSingleton==null的判断。<br/>
	 * 然后由于synchronized机制,这两个线程只有一个可以进入。然后由第一个进入的线程创建相应的实例。<br/>
	 * 如果没有第二重判断,则会创建出两个实例,这明显与单例精神不符。
	 * @return instanceof Singleton
	 */
	public static Singleton getInstance4() {
		if (mSingleton == null) {
			synchronized (Singleton.class) {
				if (mSingleton == null) {
					mSingleton = new Singleton();
				}
			}
		}
		return mSingleton;
	}

}

package com.dyk.singleton;

public class SingletonTest {

	public static void main(String[] args) {

		// hungerManTest();
		// lazyManTest();
		// synchronizedTest();
		// doubleCheckLockingTest();

	}

	private static void doubleCheckLockingTest() {
		for (int i = 0; i < 1000; i++) {
			new Thread(new Runnable() {
				@Override
				public void run() {
					Singleton mSingleton = Singleton.getInstance4();
					System.out.println("Thread name="
							+ Thread.currentThread().getName() + "实例内存地址="
							+ mSingleton.toString());
				}
			}).start();
		}
	}

	private static void synchronizedTest() {
		for (int i = 0; i < 1000; i++) {
			new Thread(new Runnable() {
				@Override
				public void run() {
					Singleton mSingleton = Singleton.getInstance2();
					System.out.println("Thread name="
							+ Thread.currentThread().getName() + "实例内存地址="
							+ mSingleton.toString());
				}
			}).start();
		}
	}

	/** 懒汉式单例测试 */
	private static void lazyManTest() {
		for (int i = 0; i < 10; i++) {
			new Thread(new Runnable() {
				@Override
				public void run() {
					// 一次测试不一定成功,需要多运行几次
					Singleton mSingleton = Singleton.getInstance();
					System.out.println("Thread name="
							+ Thread.currentThread().getName() + "实例内存地址="
							+ mSingleton.toString());
				}
			}).start();
		}
	}

	/** 饿汉式单例测试 */
	private static void hungerManTest() {
		for (int i = 0; i < 1000; i++) {
			new Thread(new Runnable() {
				@Override
				public void run() {
					Singleton mSingleton = Singleton.getInstance5();
					System.out.println("Thread name="
							+ Thread.currentThread().getName() + "实例内存地址="
							+ mSingleton.toString());
				}
			}).start();
		}
	}

}

相关文章
|
7月前
|
设计模式 缓存 安全
【设计模式】【创建型模式】单例模式(Singleton)
一、入门 什么是单例模式? 单例模式是一种设计模式,确保一个类只有一个实例,并提供一个全局访问点。它常用于需要全局唯一对象的场景,如配置管理、连接池等。 为什么要单例模式? 节省资源 场景:某些对象创
267 15
|
设计模式 安全 Java
Kotlin教程笔记(57) - 改良设计模式 - 单例模式
Kotlin教程笔记(57) - 改良设计模式 - 单例模式
117 2
|
9月前
|
设计模式 安全 Java
设计模式:单例模式
单例模式是一种创建型设计模式,确保一个类只有一个实例,并提供全局访问点。它通过私有化构造函数、自行创建实例和静态方法(如`getInstance()`)实现。适用于数据库连接池、日志管理器等需要全局唯一对象的场景。常见的实现方式包括饿汉式、懒汉式、双重检查锁、静态内部类和枚举。线程安全问题可通过`synchronized`或双重检查锁解决,同时需防止反射和序列化破坏单例。优点是避免资源浪费,缺点是可能增加代码耦合度和测试难度。实际开发中应优先选择枚举或静态内部类,避免滥用单例,并结合依赖注入框架优化使用。
|
8月前
|
设计模式 存储 安全
设计模式-单例模式练习
单例模式是Java设计模式中的重要概念,确保一个类只有一个实例并提供全局访问点。本文详解单例模式的核心思想、实现方式及线程安全问题,包括基础实现(双重检查锁)、懒汉式与饿汉式对比,以及枚举实现的优势。通过代码示例和类图,深入探讨不同场景下的单例应用,如线程安全、防止反射攻击和序列化破坏等,展示枚举实现的简洁与可靠性。
146 0
|
10月前
|
设计模式 存储 安全
设计模式2:单例模式
单例模式是一种创建型模式,确保一个类只有一个实例,并提供全局访问点。分为懒汉式和饿汉式: - **懒汉式**:延迟加载,首次调用时创建实例,线程安全通过双重检查锁(double check locking)实现,使用`volatile`防止指令重排序。 - **饿汉式**:类加载时即创建实例,线程安全但可能浪费内存。 示例代码展示了如何使用Java实现这两种模式。
241 4
|
设计模式 存储 前端开发
前端必须掌握的设计模式——单例模式
单例模式是一种简单的创建型设计模式,确保一个类只有一个实例,并提供一个全局访问点。适用于窗口对象、登录弹窗等场景,优点包括易于维护、访问和低消耗,但也有安全隐患、可能形成巨石对象及扩展性差等缺点。文中展示了JavaScript和TypeScript的实现方法。
502 13
|
设计模式 安全 Java
Kotlin教程笔记(57) - 改良设计模式 - 单例模式
Kotlin教程笔记(57) - 改良设计模式 - 单例模式
169 2
|
设计模式 Java 数据库连接
Java编程中的设计模式:单例模式的深度剖析
【10月更文挑战第41天】本文深入探讨了Java中广泛使用的单例设计模式,旨在通过简明扼要的语言和实际示例,帮助读者理解其核心原理和应用。文章将介绍单例模式的重要性、实现方式以及在实际应用中如何优雅地处理多线程问题。
205 4
|
设计模式 安全 Java
Kotlin教程笔记(57) - 改良设计模式 - 单例模式
Kotlin教程笔记(57) - 改良设计模式 - 单例模式
|
设计模式 存储 数据库连接
PHP中的设计模式:单例模式的深入理解与应用
【10月更文挑战第22天】 在软件开发中,设计模式是解决特定问题的通用解决方案。本文将通过通俗易懂的语言和实例,深入探讨PHP中单例模式的概念、实现方法及其在实际开发中的应用,帮助读者更好地理解和运用这一重要的设计模式。
133 1