zoukankan      html  css  js  c++  java
  • Java单例模式实现方式

    懒汉式-非线程安全

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

    说明:

    • 延迟加载
    • 非线程安全
    • 无法保证序列化和反序列化单例

    懒汉式-线程安全

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

    说明:

    • 延迟加载
    • 线程安全
    • 无法保证序列化和反序列化单例
    • 效率低下,synchronized很影响性能

    懒汉式-静态内部类

    public class LazyUseInner {
        private static class InnerClass {
            private static LazyUseInner instance = new LazyUseInner();
        }
        public static LazyUseInner getInstance() {
            return InnerClass.instance;
        }
        private LazyUseInner() {}
    }

    说明

    • 延迟加载
    • 线程安全
    • 无法保证序列化和反序列化单例

    懒汉式-双重锁机制

    public class DoubleLock {
        private volatile static DoubleLock instance;
        public static DoubleLock getInstance() {
            if (instance == null) { //第一次检查
                synchronized(DoubleLock.class) {
                    if (instance == null) {
                        instance = new DoubleLock();
                    }
                }
            }
            return instance;
        }
        private DoubleLock() {}
    }

    说明:

    • jdk1.5及以上线程安全,volatile语义在jdk1.5得到修正
    • 延迟加载
    • 效率不错,由于有第一步检查,规避了绝大多数执行锁同步情况
    • 无法保证序列化和反序列化单例

    饿汉式

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

    说明:

    • 简单粗暴
    • 无法延迟加载
    • 线程安全
    • 无法保证序列化和反序列化单例

    枚举单例

    public enum UseEnum {
        INSTANCE;
        public void otherMethod() {
            // do something...
        }
    }

    说明

    • 无法延迟加载
    • 线程安全
    • 保证序列化和反序列化单例
    • Effective Java中推荐使用

    结尾

    对于序列化的反序列化可以实现readResolve()来保证单例。
    项目中使用单例前最好先思考以下三点:线程安全、延迟加载、序列化与反序列化安全。

  • 相关阅读:
    8086汇编学习小记王爽汇编语言实验12
    8086汇编学习小记王爽汇编语言课程设计1
    activeMQ 持久化配置 kevin
    snmpwalk kevin
    tcp benchmark kevin
    apache camel 条件路由 kevin
    netty 并发访问测试配置 kevin
    snmp常见操作 kevin
    转发:RocketMQ与kafka的对比 kevin
    centos jdk 下载 kevin
  • 原文地址:https://www.cnblogs.com/zeling/p/8494846.html
Copyright © 2011-2022 走看看