博客
关于我
【Android】你以为的单例模式和真正的单例模式是一样的吗?
阅读量:699 次
发布时间:2019-03-17

本文共 2943 字,大约阅读时间需要 9 分钟。

单例模式深度解析:从传统到现代实践

随着软件开发的深入,单例模式被广泛用于管理类的唯一实例,确保在整个应用中只有一个对象存在。传统的单例模式虽然解决了一些问题,但在多线程环境下的表现不足,经过不断演化,逐渐提出多种更优化的实现方式。本文将从传统到现代,带你深入探讨单例模式的实现细节。

传统单例模式

传统单例模式的实现通常采用懒汉模式,具体代码如下:

public class Singleton {    private static Singleton instance;    public static Singleton getInstance() {        if (instance == null) {            instance = new Singleton();        }        return instance;    }}

这种方式通过在第一次访问时创建实例,后续直接返回现有实例,减少了内存开销。然而,这种实现在多线程环境下可能出现竞态问题,当多个线程同时获得instance == null判断时,可能导致不同的实例被创建,最终无法保证单例效果。

双重检查加锁模式:提升线程安全

为了应对多线程下的竞态问题,双重检查加锁模式被引入,代码如下:

public class Singleton {    private static Singleton instance;        public static singleton getInstance() {        if (instance == null) {            synchronized (Singleton.class) {                if (instance == null) {                    instance = new Singleton();                }            }        }        return instance;    }}

此方式首先进行非锁定检查,避免在无需时加锁,提升效率。然后在发现instance为null时,进入锁定区,确保单一线程获得锁定,创建并赋值实例。

此方案有效解决了懒汉模式在多线程环境下的线程安全问题,实现了真正的单例模式。然而,加锁运算的开销可能影响性能。

饿汉模式:优化创建时机

饿汉模式提前创建实例,代码如下:

public class Singleton {    private static Singleton instance = new Singleton();    public static Singleton getInstance() {        return instance;    }}

这种方式在类加载时就直接初始化实例,后续总是直接返回该实例。优点在于 thread safety 不需额外机制,缺点在于初次加载时可能带来启动延迟。

静态内部类:优化饿汉模式

结合饿汉模式和静态内部类的优势,实现更优的单例模式:

public class Singleton {    private Singleton() { }        public static Singleton getInstance() {        return SingletonHolder.instance;    }        private static class SingletonHolder {        private static final Singleton instance = new Singleton();    }}

此方式克服了饿汉模式的启动性能问题,同时保留了饿汉模式的 thread safety 特性。在类加载时,内部类 SingletonHolder并不会马上被初始化,只有在第一次 getInstance() 调用时才进行。这使得整体性能比饿汉模式更优。

Kotlin实现:现代化写法

Kotlin 提供了更简洁的实现方式,如懒汉模式:

class Singleton private constructor() {    companion object {        var instance : Singleton? = null            get() {                if (field == null) {                    field = Singleton()                }                return field            }        fun get(): Singleton {            return instance!!        }    }}

此外,如果需要线程安全,可以使用双重检查加锁模式:

class Singleton private constructor() {    companion object {        private var instance : Singleton? = null            get() {                if (field == null) {                    field = Singleton()                }                return field            }        @Synchronized        fun get(): Singleton {            return instance!!        }    }}

此外,Kotlin 还支持 lazy 关键字,简化代码:

class Singleton private constructor() {    companion object {        val instance by lazy (LazyThreadSafetyMode.SYNCHRONIZED) {            Singleton()        }    }}

这种方式利用Kotlin的懒汉模式,结合 thread-safety 模式,在保证单例的同时,简化了代码结构。

总结

从传统的单例模式到现代的优化实现,开发者需要根据实际场景选择合适的方式。Lazy initialization 在大多数情况下非常有效,但当需要确保 thread-safety 时,必须采取额外的同步机制。饿汉模式虽然简单,但可能导致启动延迟,而静态内部类模式则为传统饿汉模式提供了一种更优的选择。无论选择哪种实现方式,都要根据项目需求权衡性能与 thread-safety。最终,选择一个既能满足业务需求,又能在性能上的最佳平衡点,是实现高效单例模式的关键。

转载地址:http://rlnez.baihongyu.com/

你可能感兴趣的文章
Nifi同步过程中报错create_time字段找不到_实际目标表和源表中没有这个字段---大数据之Nifi工作笔记0066
查看>>
NIFI大数据进阶_FlowFile拓扑_对FlowFile内容和属性的修改删除添加_介绍和描述_以及实际操作---大数据之Nifi工作笔记0023
查看>>
NIFI大数据进阶_NIFI的模板和组的使用-介绍和实际操作_创建组_嵌套组_模板创建下载_导入---大数据之Nifi工作笔记0022
查看>>
NIFI大数据进阶_NIFI监控的强大功能介绍_处理器面板_进程组面板_summary监控_data_provenance事件源---大数据之Nifi工作笔记0025
查看>>
NIFI大数据进阶_NIFI集群知识点_认识NIFI集群以及集群的组成部分---大数据之Nifi工作笔记0014
查看>>
NIFI大数据进阶_NIFI集群知识点_集群的断开_重连_退役_卸载_总结---大数据之Nifi工作笔记0018
查看>>
NIFI大数据进阶_内嵌ZK模式集群1_搭建过程说明---大数据之Nifi工作笔记0015
查看>>
NIFI大数据进阶_外部ZK模式集群1_实际操作搭建NIFI外部ZK模式集群---大数据之Nifi工作笔记0017
查看>>
NIFI大数据进阶_离线同步MySql数据到HDFS_01_实际操作---大数据之Nifi工作笔记0029
查看>>
NIFI大数据进阶_离线同步MySql数据到HDFS_02_实际操作_splitjson处理器_puthdfs处理器_querydatabasetable处理器---大数据之Nifi工作笔记0030
查看>>
NIFI大数据进阶_连接与关系_设置数据流负载均衡_设置背压_设置展现弯曲_介绍以及实际操作---大数据之Nifi工作笔记0027
查看>>
NIFI数据库同步_多表_特定表同时同步_实际操作_MySqlToMysql_可推广到其他数据库_Postgresql_Hbase_SqlServer等----大数据之Nifi工作笔记0053
查看>>
NIFI汉化_替换logo_二次开发_Idea编译NIFI最新源码_详细过程记录_全解析_Maven编译NIFI避坑指南001---大数据之Nifi工作笔记0068
查看>>
NIFI集群_内存溢出_CPU占用100%修复_GC overhead limit exceeded_NIFI: out of memory error ---大数据之Nifi工作笔记0017
查看>>
NIFI集群_队列Queue中数据无法清空_清除队列数据报错_无法删除queue_解决_集群中机器交替重启删除---大数据之Nifi工作笔记0061
查看>>
NIH发布包含10600张CT图像数据库 为AI算法测试铺路
查看>>
Nim教程【十二】
查看>>
Nim游戏
查看>>
NIO ByteBuffer实现原理
查看>>
Nio ByteBuffer组件读写指针切换原理与常用方法
查看>>