百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 编程网 > 正文

你敢说自己了解单例模式? 单例模式的两种实现方法

yuyutoo 2024-10-12 01:23 2 浏览 0 评论

你敢说自己了解单例模式?一、背景

最近在学习设计模式,在看到单例模式的时候,我一开始以为直接很了解单例模式了,实现起来也很简单,但是实际上单例模式有着好几个变种,并且多线程中涉及到线程安全问题,那么本文我们就来好好聊聊单例模式,说一下经典三种实现方式:饿汉式、懒汉式、登记式。并且解决掉多线程中可能出现的线程安全问题。

二、基本概念

1.为什么要使用单例模式?

在我们日常的工作中,很多对象通常占用非常重要的系统资源,比如:IO处理,数据库操作等,那我们必须要限制这些对象只有且始终使用一个公用的实例,即单例。

2.单例模式的实现方式

  • 构造函数私有化,防止其他类生成唯一公用实例外的实例。且
  • 单例类应该被定义为final,也就是说单例类不能被继承,因为如果允许继承那子类就都可以创建实例,违背了类唯一实例的初衷。
  • 类中一个静态变量来保存单实例的引用。
  • 一个共有的静态方法来获取单实例的引用。

3.单例模式的UML类图

4.单例模式的经典实现方式

  • 饿汉式:一开始就创建好实例,每次调用直接返回,经典的“拿空间换时间”。
  • 懒汉式:延迟加载,第一次调用的时候才加载,然后返回,以后的每次的调用就直接返回。经典“拿时间换空间”,多线程环境下要注意解决线程安全的问题。
  • 登记式:对一组单例模式进行的维护,主要是在数量上的扩展,通过线程安全的map把单例存进去,这样在调用时,先判断该单例是否已经创建,是的话直接返回,不是的话创建一个登记到map中,再返回。

三、饿汉式---代码实现

1.单例类

package com.hafiz.designPattern.singleton;
/**
 * Desc: 单例模式-饿汉式
 * Created by hafiz.zhang on 2017/9/26.
 */
public class Singleton1 {
 // 创建全局静态变量,保证只有一个实例
 private static volatile Singleton1 instance = new Singleton1();
 private Singleton1() {
 // 构造函数私有化
 System.out.println("--调用饿汉式单例模式的构造函数--");
 }
 public static Singleton1 getInstance() {
 System.out.println("--调用饿汉式单例模式的静态方法返回实例--");
 return instance;
 }
}

2.测试类

public class DesignPatternTest {
 @Test
 public void testSingleton1() {
 System.out.println("-----------------测试饿汉式单例模式开始--------------");
 Singleton1 instance1 = Singleton1.getInstance();
 System.out.println("第二次获取实例");
 Singleton1 instance2 = Singleton1.getInstance();
 System.out.println("instance1和instance2是否为同一实例?" + (instance1 == instance2));
 System.out.println("-----------------测试饿汉式单例模式结束--------------");
 }
}

3.测试结果

四、懒汉式---代码实现

1.单例类

package com.hafiz.designPattern.singleton;
/**
 * Desc:单例模式-懒汉式
 * Created by hafiz.zhang on 2017/9/26.
 */
public class Singleton2 {
 // 创建全局静态变量,保证只有一个实例
 private static Singleton2 instance = null;
 // 构造函数私有化
 private Singleton2() {
 System.out.println("--调用懒汉式单例模式的构造方法--");
 }
 public static Singleton2 getInstance() {
 System.out.println("--调用懒汉式单例模式获取实例--");
 if (instance == null) {
 System.out.println("--懒汉式单例实例未创建,先创建再返回--");
 instance = new Singleton2();
 }
 return instance;
 }
}

2.测试类

public class DesignPatternTest {
 @Test
 public void testSingleton2() {
 System.out.println("-----------------测试懒汉式单例模式开始--------------");
 Singleton2 instance1 = Singleton2.getInstance();
 System.out.println("第二次获取实例");
 Singleton2 instance2 = Singleton2.getInstance();
 System.out.println("instance1和instance2是否为同一实例?" + (instance1 == instance2));
 System.out.println("-----------------测试懒汉式单例模式结束--------------");
 }
}

3.测试结果

细心的同学已经发现,这种实现方式,在多线程的环境中,是有线程安全安全问题的,有可能两个或多个线程判断instance都为null,然后创建了好几遍实例,不符合单例的思想,我们可以对它进行改进。

五、改进懒汉式1---代码实现

原理:使用JDK的synchronized同步代码块来解决懒汉式线程安全问题。

1.单例类

package com.hafiz.designPattern.singleton;
/**
 * Desc:单例模式-懒汉式
 * Created by hafiz.zhang on 2017/9/26.
 */
public class Singleton2 {
 // 创建全局静态变量,保证只有一个实例
 private static Singleton2 instance = null;
 // 构造函数私有化
 private Singleton2() {
 System.out.println("--调用懒汉式单例模式的构造方法--");
 }
 public static Singleton2 getInstance() {
 System.out.println("--调用懒汉式单例模式获取实例--");
     if (instance != null) {
        System.out.println("--懒汉式单例实例已经创建,直接返回--");
  return instance;
     }
 synchronized (Singleton2.class) {
    if (instance == null) {
   System.out.println("--懒汉式单例实例未创建,先创建再返回--");
   instance = new Singleton2();
    }
 }
 return instance;
 }
} 

2.测试结果

六、改进懒汉式2---代码实现

原理:使用JVM隐含的同步和类级内部类来解决,JVM隐含的同步解决了多线程情况下线程安全的问题,类级内部类解决只有使用的时候才加载(延迟加载)的问题。

1.JVM隐含的同步有哪些?

  • 静态初始化器(在静态字段上或static{}静态代码块的初始化器)初始化数据时
  • 访问final字段时
  • 在创建线程之前创建对象时
  • 线程可以看见它将要处理的对象时

2.什么是类级内部类?

  • 有static修饰的成员式内部类。没有static修饰的成员式内部类叫对象级内部类。
  • 类级内部类相当于其外部类的static成分,他的对象与外部类对象间不存在依赖关系,因此可直接创建,而对象级内部类的实例,是绑定在外部对象实例中的。
  • 类级内部类中,可以定义静态的方法。在静态的方法中只能够引用外部类的中的静态成员方法或者成员变量
  • 类级内部类相当于其外部类的成员,只有在第一次被使用的时候才会被装载

3.单例类

package com.hafiz.designPattern.singleton;
/**
 * Desc:单例模式-改进懒汉式
 * Created by hafiz.zhang on 2017/9/26.
 */
public class Singleton3 {
 private static class Singleton4 {
 private static Singleton3 instance;
 static {
 System.out.println("--类级内部类被加载--");
 instance = new Singleton3();
 }
 private Singleton4() {
 System.out.println("--调用类级内部类的构造函数--");
 }
 }
 private Singleton3() {
 System.out.println("--调用构造函数--");
 }
 public static Singleton3 getInstance() {
 System.out.println("--开始调用共有方法返回实例--");
 Singleton3 instance;
 System.out.println("---------------------------");
 instance = Singleton4.instance;
 System.out.println("返回单例");
 return instance;
 }
}

4.测试类

package com.hafiz.www;
import com.hafiz.designPattern.observer.ConcreteObserver;
import com.hafiz.designPattern.observer.ConcreteSubject;
import com.hafiz.designPattern.singleton.Singleton1;
import com.hafiz.designPattern.singleton.Singleton2;
import com.hafiz.designPattern.singleton.Singleton3;
import com.hafiz.designPattern.singleton.Singleton4;
import com.hafiz.designPattern.singleton.Singleton4Child1;
import com.hafiz.designPattern.singleton.SingletonChild2;
import org.junit.Test;
/**
 * Desc:设计模式demo单元测试类
 * Created by hafiz.zhang on 2017/7/27.
 */
public class DesignPatternTest {
 @Test
 public void testSingleton3() {
 System.out.println("-----------------测试改进懒汉式单例模式开始--------------");
 Singleton3 instance1 = Singleton3.getInstance();
 System.out.println("第二次获取实例");
 Singleton3 instance2 = Singleton3.getInstance();
 System.out.println("instance1和instance2是否为同一实例?" + (instance1 == instance2));
 System.out.println("-----------------测试改进懒汉式单例模式结束--------------");
 }
}

5.测试结果

七、登记式--代码实现

1.基类

package com.hafiz.designPattern.singleton;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
/**
 * Desc: 单例模式-登记式
 * Created by hafiz.zhang on 2017/9/26.
 */
public class Singleton4 {
 private static Map<String, Singleton4> map = new ConcurrentHashMap<>();
 protected Singleton4() {
 System.out.println("--私有化构造函数被调用--");
 }
 public static Singleton4 getInstance(String name) {
 if (name == null) {
 name = Singleton4.class.getName();
 System.out.println("--name为空,默认赋值为:--" + Singleton4.class.getName());
 }
 if (map.get(name) != null) {
 System.out.println("name对应的值存在,直接返回");
 return map.get(name);
 }
 System.out.println("name对应的值不存在,先创建,再返回");
 try {
 Singleton4 result = (Singleton4)Class.forName(name).newInstance();
 map.put(name, result);
 return result;
 } catch (InstantiationException e) {
 e.printStackTrace();
 } catch (IllegalAccessException e) {
 e.printStackTrace();
 } catch (ClassNotFoundException e) {
 e.printStackTrace();
 }
 return null;
 }
 public Map<String, Singleton4> getMap() {
 return map;
 }
}

2.子类1

package com.hafiz.designPattern.singleton;
/**
 * Desc:
 * Created by hafiz.zhang on 2017/9/26.
 */
public class Singleton4Child1 extends Singleton4 {
 public static Singleton4Child1 getInstance() {
 return (Singleton4Child1) Singleton4.getInstance("com.hafiz.designPattern.singleton.Singleton4Child1");
 }
}

3.子类2

package com.hafiz.designPattern.singleton;
/**
 * Desc:
 * Created by hafiz.zhang on 2017/9/26.
 */
public class SingletonChild2 extends Singleton4 {
 public static SingletonChild2 getInstance() {
 return (SingletonChild2) Singleton4.getInstance("com.hafiz.designPattern.singleton.SingletonChild2");
 }
}

4.测试类

public class DesignPatternTest {
 @Test
 public void testSingleton4() {
 System.out.println("-----------------测试登记式单例模式开始--------------");
 System.out.println("第一次取得实例");
 Singleton4 instance1 = Singleton4.getInstance(null);
 System.out.println("res:" + instance1);
 System.out.println("第二次获取实例");
 Singleton4Child1 instance2 = Singleton4Child1.getInstance();
 System.out.println("res:" + instance2);
 System.out.println("第三次获取实例");
 SingletonChild2 instance3 = SingletonChild2.getInstance();
 System.out.println("res:" + instance3);
 System.out.println("第四次获取实例");
 SingletonChild2 instance4 = new SingletonChild2();
 System.out.println("res:" + instance4);
 System.out.println("输出父类Map中所有的单例");
 Map<String, Singleton4> map = instance1.getMap();
 for (Map.Entry<String, Singleton4> item : map.entrySet()) {
 System.out.println("map-item:" + item.getKey() + "=" + item.getValue());
 }
 System.out.println("instance1和instance2是否为同一实例?" + (instance1 == instance2));
 System.out.println("-----------------测试登记式单例模式结束--------------");
 }
}

5.测试结果

该解决方案的缺点:基类的构造函数对子类公开了(protected),有好的解决方案的博友可以讨论指教~

八、总结

经过本文,我们就搞明白了什么叫单例模式,如何优雅的实现经典的单例模式,如何进行拓展和开发具有线程安全的单例模式。对于我们以后的开发非常有帮助,也让我们更加了解单例模式。

相关推荐

史上最全的浏览器兼容性问题和解决方案

微信ID:WEB_wysj(点击关注)◎◎◎◎◎◎◎◎◎一┳═┻︻▄(页底留言开放,欢迎来吐槽)●●●...

平面设计基础知识_平面设计基础知识实验收获与总结
平面设计基础知识_平面设计基础知识实验收获与总结

CSS构造颜色,背景与图像1.使用span更好的控制文本中局部区域的文本:文本;2.使用display属性提供区块转变:display:inline(是内联的...

2025-02-21 16:01 yuyutoo

写作排版简单三步就行-工具篇_作文排版模板

和我们工作中日常word排版内部交流不同,这篇教程介绍的写作排版主要是用于“微信公众号、头条号”网络展示。写作展现的是我的思考,排版是让写作在网格上更好地展现。在写作上花费时间是有累积复利优势的,在排...

写一个2048的游戏_2048小游戏功能实现

1.创建HTML文件1.打开一个文本编辑器,例如Notepad++、SublimeText、VisualStudioCode等。2.将以下HTML代码复制并粘贴到文本编辑器中:html...

今天你穿“短袖”了吗?青岛最高23℃!接下来几天气温更刺激……

  最近的天气暖和得让很多小伙伴们喊“热”!!!  昨天的气温到底升得有多高呢?你家有没有榜上有名?...

CSS不规则卡片,纯CSS制作优惠券样式,CSS实现锯齿样式

之前也有写过CSS优惠券样式《CSS3径向渐变实现优惠券波浪造型》,这次再来温习一遍,并且将更为详细的讲解,从布局到具体样式说明,最后定义CSS变量,自定义主题颜色。布局...

柠檬科技肖勃飞:大数据风控助力信用社会建设

...

你的自我界限够强大吗?_你的自我界限够强大吗英文

我的结果:A、该设立新的界限...

行内元素与块级元素,以及区别_行内元素和块级元素有什么区别?

行内元素与块级元素首先,CSS规范规定,每个元素都有display属性,确定该元素的类型,每个元素都有默认的display值,分别为块级(block)、行内(inline)。块级元素:(以下列举比较常...

让“成都速度”跑得潇潇洒洒,地上地下共享轨交繁华
让“成都速度”跑得潇潇洒洒,地上地下共享轨交繁华

去年的两会期间,习近平总书记在参加人大会议四川代表团审议时,对治蜀兴川提出了明确要求,指明了前行方向,并带来了“祝四川人民的生活越来越安逸”的美好祝福。又是一年...

2025-02-21 16:00 yuyutoo

今年国家综合性消防救援队伍计划招录消防员15000名

记者24日从应急管理部获悉,国家综合性消防救援队伍2023年消防员招录工作已正式启动。今年共计划招录消防员15000名,其中高校应届毕业生5000名、退役士兵5000名、社会青年5000名。本次招录的...

一起盘点最新 Chrome v133 的5大主流特性 ?

1.CSS的高级attr()方法CSSattr()函数是CSSLevel5中用于检索DOM元素的属性值并将其用于CSS属性值,类似于var()函数替换自定义属性值的方式。...

竞走团体世锦赛5月太仓举行 世界冠军杨家玉担任形象大使

style="text-align:center;"data-mce-style="text-align:...

学物理能做什么?_学物理能做什么 卢昌海

作者:曹则贤中国科学院物理研究所原标题:《物理学:ASourceofPowerforMan》在2006年中央电视台《对话》栏目的某期节目中,主持人问过我一个的问题:“学物理的人,如果日后不...

你不知道的关于这只眯眼兔的6个小秘密
你不知道的关于这只眯眼兔的6个小秘密

在你们忙着给熊本君做表情包的时候,要知道,最先在网络上引起轰动的可是这只脸上只有两条缝的兔子——兔斯基。今年,它更是迎来了自己的10岁生日。①关于德艺双馨“老艺...

2025-02-21 16:00 yuyutoo

取消回复欢迎 发表评论: