创建型模式-单例模式

介绍

  • 使用意图:<h6>让一个类只有一个实例化对象。

  • 主要解决:<h6>一个全局使用的类频繁地创建与销毁。

  • 何时使用:<h6>想控制new实例数目,节省系统资源的时候。

  • 如何解决:<h6>判断系统是否已经有这个单例,如果有则返回,如果没有则创建。

  • 关键代码:<h6>构造函数是私有的,并且提供一个获取单例的方法。
  • 应用实例:
    • <h6>case:一个党只能有一个主席。
    • <h6>case:多线程操作文件的时候,不可避免多个线程操作一个文件,文件的处理必须通过唯一的实例来进行
      • <h5>优点:
        <h6>1、在内存里只有一个实例,减少了内存的开销,尤其是频繁的创建和销毁实例。
        <h6>2、避免对资源的多重占用(比如写文件操作)。
      • <h5>缺点:<h6>1、没有接口,不能继承,与单一职责原则冲突,一个类应该只关心内部逻辑,而不关心外面怎么样来实例化。
      • <h5>场景:
        <h6>1、要求生产唯一序列号。
        <h6>2、WEB 中的计数器,不用每次刷新都在数据库里加一次,用单例先缓存起来。
        <h6>3、创建的一个对象需要消耗的资源过多,比如 I/O 与数据库的连接等。

实现


<h6>1、懒汉式,线程安全
<pre>
//1、懒汉式,线程安全
//2、Lazy 初始化,多线程安全
//3、synchronized 保证了线程安全,但是效率很低,99%不需要加锁。
public class SingleObjectLazy {
private static SingleObjectLazy instance = new SingleObjectLazy();
private SingleObjectLazy (){}
public static synchronized SingleObjectLazy getInstance() {
return instance;
}
}
</pre>
<h6>2、饿汉式
<pre>
// 1、饿汉式,线程安全
// 2、不是Lazy 初始化,没有加锁,执行效率会提高
// 3、它基于 classloder 机制避免了多线程的同步问题(推荐)
public class SingleObjectHungry {
private static SingleObjectHungry instance;
private SingleObjectHungry() {
}
public static SingleObjectHungry getInstance() {
if (instance == null) {
instance = new SingleObjectHungry();
}
return instance;
}
}</pre>
<h6>3、双检锁/双重校验锁
<pre>
// 1、双检锁/双重校验锁(DCL,即 double-checked locking)
// 2、是Lazy 初始化,安全且在多线程情况下能保持高性能。
// 3、volatile 要求JVM保证线程改变 x 后能立即应用到所有线程
public class SingleObjectDcl {
private volatile static SingleObjectDcl singleton;
private SingleObjectDcl() {}
public static SingleObjectDcl getSingleton() {
if (singleton == null) {
synchronized (SingleObjectDcl.class) {
if (singleton == null) {
singleton = new SingleObjectDcl();
}
}
}
return singleton;
}
}</pre>
<h6>4、静态内部类
<pre>
// 1、静态内部类
// 2、不是Lazy 初始化,利用了 classloder 机制来保证初始化 instance 时只有一个线程
// 3、和饿汉区别在,SingleObjectStatic装载后INSTANCE不一定初始化,只有显示调用getInstance在被实例化
public class SingleObjectStatic {
private static class SingletonHolder {
private static final SingleObjectStatic INSTANCE = new SingleObjectStatic();
}
private SingleObjectStatic (){}
public static final SingleObjectStatic getInstance() {
return SingletonHolder.INSTANCE;
}
}</pre>


<h6>小结:单例模式getInstance() 方法中需要使用同步锁 synchronized (Singleton.class) 防止多线程同时进入造成 instance 被多次实例化,如果没有同步(synchronized)线程安全算不上一个单例模式。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

  • 1 场景问题# 1.1 读取配置文件的内容## 考虑这样一个应用,读取配置文件的内容。 很多应用项目,都有与应用相...
    七寸知架构阅读 11,787评论 12 68
  • 前言 本文主要参考 那些年,我们一起写过的“单例模式”。 何为单例模式? 顾名思义,单例模式就是保证一个类仅有一个...
    tandeneck阅读 7,251评论 1 8
  • 单例模式(Singleton Pattern)是众多设计模式中较为简单的一个,同时它也是面试时经常被提及的问题,如...
    廖少少阅读 3,725评论 0 1
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 33,345评论 18 399
  • 最近正在读吴晓波的《激荡三十年1978-2008》,也许是褪去了年轻激情,我早已经不喜欢读风华雪月的小说了,现在读...
    张sara阅读 2,271评论 0 0

友情链接更多精彩内容