设计模式之单例模式

发表于:2017-5-23 13:39

字体: | 上一篇 | 下一篇 | 我要投稿

 作者:MVBin    来源:简书

  单例模式(Singleton Pattern)
  程序设计过程中,当我们需要一个全局的,独一无二的实例时,我们可以使用单例模式进行设计。
  一般写法(线程不安全)
//一个普通的类
public class Singleton {
//.....该类其他的一些属性
//uniqueInstance:确保全局只有一个实例
private static Singleton uniqueInstance;
//私有化的构造方法:只能在类的内部创建实例
private Singleton() {
}
//静态方法getInstance():外部类可以通过Singleton.getInstance()得到一个实例
public static Singleton getInstance() {
//uniqueInstance,没有的话创建一个,此处使用的是懒汉式加载,只在需要时候进行创建
if(uniqueInstance == null) {
uniqueInstance = new Singleton();
}
return uniqueInstance;
}
//.......该类其他的一些方法
}
  如果该类在应用程序并不是关键程序,不经常使用,这种写法的单例模式就足够,但是,如果该类很重要,需经常使用,则这种写法会导致多线程问题。
  同步getInstance()方法(线程安全)
//一个普通的类
public class Singleton {
//.....该类其他的一些属性
//uniqueInstance:确保全局只有一个实例
private static Singleton uniqueInstance;
//私有化的构造方法:只能在类的内部创建实例
private Singleton() {
}
//静态方法getInstance():外部类可以通过Singleton.getInstance()得到一个实例
public static synchronized Singleton getInstance() {
//uniqueInstance,没有的话创建一个,此处使用的是懒汉式加载,只在需要时候进行创建
if(uniqueInstance == null) {
uniqueInstance = new Singleton();
}
return uniqueInstance;
}
//.......该类其他的一些方法
}
  使用synchronized关键字修饰getInstance()方法,这样就迫使每个线程进入这个方法之前,要先等到别的线程离开这个方法。但是同步的方式会降低性能,在每次调用这个方法时,都要同步,其实,真正需要的时只是在第一次创建实例时进行同步的,因此,这种写法不可取。
  饿汉式写法(线程安全)
//一个普通的类
public class Singleton {
//.....该类其他的一些属性
//首先定义一个静态变量uniqueInstance,确保全局只有一个实例,另外,在JVM加载这个类的时候,就直接对该类进行实例化(饿汉式),以此避免多线程问题
private static Singleton uniqueInstance = new Singleton();
//私有化的构造方法:只能在类的内部创建实例
private Singleton() {
}
//静态方法getInstance():外部类可以通过Singleton.getInstance()得到一个实例
public static Singleton getInstance() {
return uniqueInstance;
}
//.......其他该类的一些方法
}
  这种写法可以有效避免多线程问题,但是假如这个类加载时非常耗用资源,程序启动效率低下,就不要用这种写法了。
  双重检查加锁写法(线程安全)
//一个普通的类
public class Singleton {
//.....该类其他的一些属性
//volatile关键字适用于jdk1.5以上版本,其确保当uniqueInstance变量被实例化时,多个线程正确处理uniqueInstance变量
private volatile static Singleton uniqueInstance;
//私有化构造方法
private Singleton() {
}
//只有在第一次创建实例时,才会彻底执行该方法
public static Singleton getInstance() {
//第一次检查是否为null
if(uniqueInstance == null) {
//同步区块
synchronized (Singleton.class) {
//第二次检查是否为null
if(uniqueInstance == null) {
//这才创建实例
uniqueInstance = new Singleton();
}
}
}
return uniqueInstance;
}
//.......其他该类的一些方法
}
  这种写法提升了性能,但是须在jdk1.5版本以上使用
  枚举类型写法(线程安全)
//一个普通的枚举类型
public enum SingletonEnum {
//调用SingletonEnum.INSTANCE即可获得一个实例,枚举类型自动提供了序列化机制,绝对防止多次实例化
INSTANCE;
//...其他方法
}
  需要jdk1.5版本以上,采用单元素的枚举类型的方式,比较简洁,推荐采用这种写法。
《2023软件测试行业现状调查报告》独家发布~

关注51Testing

联系我们

快捷面板 站点地图 联系我们 广告服务 关于我们 站长统计 发展历程

法律顾问:上海兰迪律师事务所 项棋律师
版权所有 上海博为峰软件技术股份有限公司 Copyright©51testing.com 2003-2024
投诉及意见反馈:webmaster@51testing.com; 业务联系:service@51testing.com 021-64471599-8017

沪ICP备05003035号

沪公网安备 31010102002173号