• Java单例模式的5种实现方式


    1.饿汉式、不支持并发:

    package com.ou;
    
    //饿汉式
    public class Singleton1 {
        private Singleton1() {
        }
        private static Singleton1 instance = new Singleton1();
        
        public static Singleton1 getInstance(){
            return instance;
        }
    
    }

     此模式只能运行在单线程下,且类在加载时就已经创建好了实例,不管需不需要用。

    2.懒汉式、不支持并发:

    package com.ou;
    
    //懒汉式、不支持多并发
    public class Singleton2 {
        private Singleton2() {
        }
        private static Singleton2 instance = null;
        public static Singleton2 getInstance(){
            if (instance == null){
                instance = new Singleton2();
            }
            return instance;
        }
    
    }

      此模式只能运行在单线程下,在调用获取实例的方法时才创建实例。

    3.懒汉式、支持并发、synchronized:

    package com.ou;
    
    //懒汉式、支持多并发、效率低
    public class Singleton3 {
        private Singleton3() {
        }
        private static Singleton3 instance = null;
        public synchronized static Singleton3 getInstance(){
            if (instance == null){
                instance = new Singleton3();
            }
            return instance;
        }
    
    }
    synchronized 锁住了整个方法,当有多个线程需要访问方法时,不管实例有没有创建,都会要排队等待才能拿到实例,效率低。
    需要改进:只有第一次创建实例时才需要锁,其他时候不需要加锁。

    4.双重检查锁 、volatile(常用):

    package com.ou;
    
    //double checked locking、支持多并发、效率高、添加volatile关键字
    public class Singleton4 {
        private Singleton4() {
        }
        private volatile static Singleton4 instance = null;
        public static Singleton4 getInstance(){
            if (instance == null){//1
                synchronized (Singleton4.class) {
                    if (instance == null)//2
                        instance = new Singleton4();
                }
            }
            return instance;
        }
    }
    volatile 关键字保证了内存可见性,所有线程都会去主存中取数据而不是在线程的缓存中取,保证了数据的更新能实时地对任何线程可见。
    假如有两个线程同时到达了1,它们都去创建实例,这时候如果没有第二次判断,就会多次创建实例了。二次判断保证了多线程下只创建一个实例。

    5.静态私有内部类(常用):

    package com.ou;
    
    //静态私有内部类、支持多并发、效率高、
    public class Singleton5 {
        private Singleton5() {
        }
        private static class SingletonHolder{
            private static Singleton5 instance = new Singleton5();
        }
        public static Singleton5 getInstance(){
            return SingletonHolder.instance;
        }
    }

    内部类的好处:内部类在被调用的时候才实例化其静态成员变量,高!

  • 相关阅读:
    apply和call详解
    this用法
    jquery的each()详细介绍
    摹客食堂|你真的会做交付文档吗?
    资源分享|找不到素材?这份免费APP UI Kit资源大礼包送给你!
    功能播报|PRD可以在线审阅啦!让文档管理更轻松~
    工具推荐|2019年UI设计师必备工具清单
    案例分析|最佳倒数计时器设计分析【附原型实例】
    摹客食堂|新"葵花宝典"--用户体验设计知识大全
    灵感专题|2019年优秀网页设计作品赏析#9月
  • 原文地址:https://www.cnblogs.com/tongkey/p/8377702.html
Copyright © 2020-2023  润新知