单例的几种实现方式

单例

一、相关概念

单例:是java23中常见的设计模式之一,属于创建型模式,保证一个类只有一个实例,并对外提供调用对象该实例的方法。

意图:保证一个类在应用全局只有一个对象,减少对象的品频繁创建。

何时使用:控制实例数量,减少对象的创建,优化系统资源。

关键代码:1、构造函数私有;2、单例使用静态引用。

二、实现方式

从实例初始化的时机可以分为饿汉式、懒汉式。

饿汉式:资源加载时实例就被初始化,一般是在借用类加载机制,在类加载的初始化单例对象,可以避免多线程同步问题,但是容易产生系统垃圾。

懒汉式:第一次需要单例对象时调用单例的实现方法,可以避免内存浪费。

从线程安全性角度来说,可以分为单线程、多线程安全。

还有业内通用的DCL模式,即双检锁/双重校验锁(DCL,即 double-checked locking)。

1、懒汉式,线程不安全

该方法只适用于单线程,若是在多线程情况下,多线程走到instance==null,可能生成多个实例。

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

        public static void main(String[] args) {
            Singleton1 instance1 = getInstance();
            Singleton1 instance2 = getInstance();
            log.info("实例对象是否相等: "+instance1.equals(instance2));
        }
    }

2、懒汉式,线程安全

该方法可用于多线程,懒汉式加载,第一次调用时初始化对象,节省内存,使用synchronized保证对象唯一,但是加锁印象到效率

 static class Singleton2 {
        private static Singleton2 instance;
        private Singleton2(){};
        public static synchronized Singleton2 getInstance(){
            if (instance==null){
                instance=new Singleton2();
            }
            return instance;
        }
        public static void main(String[] args) {
            Singleton2 instance1 = getInstance();
            Singleton2 instance2 = getInstance();
            log.info("实例对象是否相等: "+instance1.equals(instance2));
        }
    }

3、饿汉式

该方法用静态方法修饰并初始化变量,利用类加载机制保证对象的唯一性。但是属于饿汉式,不是懒加载,浪费内存,好处是容易理解。

   static class Singleton3 {
        private static Singleton3 instance=new Singleton3();
        private Singleton3(){}
        private static Singleton3 getInstance(){
            return instance;
        }
        public static void main(String[] args) {
            Singleton3 instance1 = getInstance();
            Singleton3 instance2 = getInstance();
            log.info("实例对象是否相等: "+instance1.equals(instance2));
        }
    }

4、双检锁/双重校验锁(DCL,即 double-checked locking)

业界广泛,面试常问的单例模式。

好处:1、第一次校验可以减少加锁的次数,提高使用效率,在多线程下保证高性能;

​ 2、懒汉式加载,只有在第一次调用时初始化对象,节省内存;

​ 3、使用volatile,利用其禁止指令重排的特性,在初始化instance对象时可以保证强唯一性。

static class Singleton4 {
        private volatile static Singleton4 instance;
        private Singleton4(){}
        public static Singleton4 getInstance(){
            if (instance==null){
                synchronized (Singleton4.class){
                    if (instance==null){
                        instance=new Singleton4();
                    }
                }
            }
            return instance;
        }
        public static void main(String[] args) {
            Singleton4 instance1 = getInstance();
            Singleton4 instance2 = getInstance();
            log.info("实例对象是否相等: "+instance1.equals(instance2));
        }
    }

5、登记式/静态内部类

特征和好处:1、首先使用静态方法保证对象的唯一性;

​ 2、内部类的使用又保证在第一次调用时初始化对象,节省内存;

static class Singleton5 {
        private static class Singleton5Holder{
            private static final Singleton5 SINGLETON_5=new Singleton5();
        }
        public static Singleton5 getInstance(){
            return Singleton5Holder.SINGLETON_5;
        }
        public static void main(String[] args) {
            Singleton5 instance1 = getInstance();
            Singleton5 instance2 = getInstance();
            log.info("实例对象是否相等: "+instance1.equals(instance2));
        } 
    }

6、枚举

枚举是jdk1.5之后推出,该单例实现方式在业内使用的并不频繁,但是该方法可以避免反序列化重新创建对象,可以保证绝对唯一性。

    enum  Singleton6 {
        INSTANCE;
        public static Singleton6 getInstance(){
            return INSTANCE;
        }
        public static void main(String[] args) {
            Singleton6 instance1 = getInstance();
            Singleton6 instance2 = getInstance();
            log.info("实例对象是否相等: "+instance1.equals(instance2));
        }
    }

总结:一般情况下,不建议使用第 1 种和第 2 种懒汉方式,建议使用第 3 种饿汉方式。只有在要明确实现 lazy loading 效果时,才会使用第 5 种登记方式。如果涉及到反序列化创建对象时,可以尝试使用第 6 种枚举方式。如果有其他特殊的需求,可以考虑使用第 4 种双检锁方式。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,362评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,330评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,247评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,560评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,580评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,569评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,929评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,587评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,840评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,596评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,678评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,366评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,945评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,929评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,165评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,271评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,403评论 2 342