java 单例模式 懒汉和饿汉

单线程下的单例模式

定义:

确保一个类只有一个实例,并提供一个全局访问点。

单例代码:

package wfb.pattern;


public class Singleton {
	private static Singleton instance;
	private Singleton() {
		System.out.println("实例化了一个新的单例");
	}
	public static Singleton getInstance() {
		if(instance == null) {
			instance = new Singleton();
		}
		return instance;
	}
}

测试代码:

package wfb.patthernTest;

import org.junit.Test;

import wfb.pattern.Singleton;

public class SingletonTest {
	@Test
	public void testSinglon1() {
		Singleton singlon1 = Singleton.getInstance();
		Singleton singlon2 = Singleton.getInstance();
	}
	@Test
	public void testSingleton2() throws Exception {
		while(true) {
			new Thread(new Runnable() {
				public void run() {
					try {
						Thread.sleep(10);
					} catch (InterruptedException e) {
						e.printStackTrace();
					}
					Singleton singlon = Singleton.getInstance();
				}
			}).start();
		}
	}
}

测试结果:

单线程 :
单线程下,只会有一个实例
多线程 :
在多线程的环境下,简单的单例模式生成了多个实例

原因分析:

    如果多个线程同时走到if(instance == null)的判断语句,会因为还未创建实例而进入到实例化代码,从而导致了单件被实例化多次。换句话说,根本原因是if(instance == null) instance = new Singleton()这两步操作不是原子性的。
为此,我们有两种解决方法:
1.在类加载后就初始化instance,之后就不在需要创建实例了。
2.用同步包起这两步操作,从而使得它具有原子性。
懒汉和饿汉模式就这两种方法分别给出了多线程下单例模式的解决方案。

多线程下的单例模式

饿汉代码:

package wfb.pattern;
//饿汉模式通过提前初始化一个实例来解决多线程问题
public class SingletonHungry {
	private static SingletonHungry instance = new SingletonHungry();
	private SingletonHungry() {
		System.out.println("实例化了一个新的单例");
	}
	public static SingletonHungry getInstance() {
		return instance;
	}
}

饿汉测试:

@Test
	public void testSingletonHungry() throws Exception{
		while(true) {
			new Thread(new Runnable() {
				public void run() {
					try {
						Thread.sleep(10);
					} catch (InterruptedException e) {
						e.printStackTrace();
					}
					SingletonHungry singletonHungry = SingletonHungry.getInstance();
				}
			}).start();
		}
	}

饿汉结果:

只执行了一次实例化

懒汉代码:

package wfb.pattern;
//懒汉模式通过同步解决多线程问题。
public class SingletonLazy {
	private static SingletonLazy instance;
	private SingletonLazy() {
		System.out.println("实例化了一个新的单例");
	}
	public static SingletonLazy getInstance() {
		if(instance != null) return instance;//为了增加效率,当拥有了实例后,我们直接返回实例。因为拥有了实例后,状态就跟饿汉一致了,显然不会发生线程问题。
		synchronized (SingletonLazy.class) {//因为是静态方法,所以用类做同步锁
			if(instance == null) instance = new SingletonLazy();
			return instance;
		}
	}
}

懒汉测试:

@Test
	public void testSingletonLazy() throws Exception{
		while(true) {
			new Thread(new Runnable() {
				public void run() {
					try {
						Thread.sleep(10);
					} catch (InterruptedException e) {
						e.printStackTrace();
					}
					SingletonLazy singletonLazy = SingletonLazy.getInstance();
				}
			}).start();
		}
	}

懒汉结果:

只执行了一次实例化

  • 6
    点赞
  • 26
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
单例模式是一种设计模式,用于确保一个类只有一个实例,并提供一个全局访问点来获取该实例。在单例模式中,懒汉饿汉是两种常见的实现方式。 1. 懒汉模式懒汉模式是指在需要使用实例时才创建对象。以下是懒汉模式的代码示例: ```java public class Singleton { private static Singleton instance; private Singleton() { // 私有构造函数 } public static synchronized Singleton getInstance() { if (instance == null) { instance = new Singleton(); } return instance; } } ``` 在懒汉模式中,通过一个静态变量`instance`来保存实例,初始值为null。在`getInstance()`方法中,首先检查`instance`是否为null,如果为null,则创建一个新的实例并赋值给`instance`,然后返回该实例。由于懒汉模式是线程安全的,因此在`getInstance()`方法上加了`synchronized`关键字,确保在多线程环境下只有一个线程可以创建实例。 2. 饿汉模式饿汉模式是指在类加载时就创建对象,无论是否使用该实例。以下是饿汉模式的代码示例: ```java public class Singleton { private static Singleton instance = new Singleton(); private Singleton() { // 私有构造函数 } public static Singleton getInstance() { return instance; } } ``` 在饿汉模式中,通过一个静态变量`instance`来保存实例,并在定义时就进行初始化。在`getInstance()`方法中,直接返回`instance`即可。由于饿汉模式在类加载时就创建了实例,因此不存在线程安全问题。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值