Effective Java - 构造器私有、枚举和单例

简介: Singleton 是指仅仅被实例化一次的类。Singleton代表了无状态的对象像是方法或者本质上是唯一的系统组件。使类称为Singleton 会使它的客户端测试变得十分困难。因为不可能给Singleton替换模拟实现。除非实现一个充当其类型的接口

Effective Java - 构造器私有、枚举和单例


[TOC]


Singleton 是指仅仅被实例化一次的类。Singleton代表了无状态的对象像是方法或者本质上是唯一的系统组件。使类称为Singleton 会使它的客户端测试变得十分困难。因为不可能给Singleton替换模拟实现。除非实现一个充当其类型的接口


饿汉式单例


静态常量


下面有两种方法实现一个单例,两者都基于保持构造器私有并且导出一个公有的静态成员提供一个唯访问该实例的入口。在第一种方法中,这个成员的属性是 final

// 提供属性是公有的、唯一的单例
public class Elvis {
  public static final Elvis INSTANCE = new Elvis();
  private Elvis();
  public void leaveTheBuilding();
}

这是一个饿汉式的实现。这个私有的构造器仅仅被调用一次,因为Elvis 是  static final 的,所以 INSTANCE 是一个常量,编译期间进行初始化,并且值只能被初始化一次,致使 INSTANCE 不能再指向任意其他的对象,没有任何客户端能够改变这个结果。但是需要注意一点:有特权的客户端能够使用反射中的 AccessibleObject.setAccessible 访问私有的构造器。为了防御这种攻击,把构造器修改为在第二次实例化的时候抛出异常。见如下的例子

public class Elvis {
    static boolean flag = false;
    private Elvis(){
        if(flag == false) {
            flag = !flag;
        }
        else {
            throw new RuntimeException("单例模式被侵犯!");
        }
    }
    public static class SingletonHolder {
        private static final Elvis INSTANCE = new Elvis();
    }
    public static Elvis getInstance(){
        return SingletonHolder.INSTANCE;
    }
    public static void main(String[] args) throws Exception {
        Class<Elvis> el = Elvis.class;
        // 获得无参数私有的构造器
        Constructor<Elvis> constructor = el.getDeclaredConstructor();
        // 暴力破解private 私有化
        constructor.setAccessible(true);
        // 生成新的实例
        Elvis elvis = constructor.newInstance();
        Elvis instance = Elvis.getInstance();
        System.out.println(elvis == instance);
    }
}
Exception in thread "main" java.lang.ExceptionInInitializerError
  at effectiveJava.effective03.Elvis.getInstance(Elvis.java:22)
  at effectiveJava.effective03.Elvis.main(Elvis.java:33)
Caused by: java.lang.RuntimeException: 单例模式被侵犯!
  at effectiveJava.effective03.Elvis.<init>(Elvis.java:13)
  at effectiveJava.effective03.Elvis.<init>(Elvis.java:5)
  at effectiveJava.effective03.Elvis$SingletonHolder.<clinit>(Elvis.java:18)
  ... 2 more

注释掉利用反射获取私有构造函数的代码,发现instance实例可以正常输出

Elvis instance = Elvis.getInstance();
System.out.println(instance);

console: effectiveJava.effective03.Elvis@266474c2


在实现Singleton 的第二种方法中,公有的成员是个静态方法

public class ElvisSingleton {
    private static final ElvisSingleton INSTANCE = new ElvisSingleton();
    private ElvisSingleton(){}
    public static ElvisSingleton newInstance(){
        return INSTANCE;
    }
    public void leaveBuilding(){}
}

对于静态方法 newInstance 说所有的调用,都会返回一个INSTANCE对象,所以,永远不会创建其他 ElvisSingleton 实例


公有属性最大的优势在于能够很清楚的描述类是单例的:公有的属性是final的,所以总是能够包含相同的对象引用。第二个优势就是就是比较简单。


静态代码块


静态代码块是静态常量的变种,就是把静态常量的初始化放在了静态代码块中解析,初始化。读者可能对这种方式产生疑惑,请详见


类加载机制 https://blog.csdn.net/ns_code/article/details/17881581

public class ElvisStaticBlock {
    private static final ElvisStaticBlock block;
    static {
        block = new ElvisStaticBlock();
    }
    private ElvisStaticBlock(){}
    public static ElvisStaticBlock newInstance(){
        return block;
    }
}

优点:这种写法比较简单,就是在类装载的时候就完成实例化。避免了线程同步问题。


缺点:在类装载的时候就完成实例化,没有达到Lazy Loading的效果。如果从始至终从未使用过这个实例,则会造成内存的浪费。


懒汉式单例


与饿汉式对应的就是懒汉式,这两者都是属于单例模式的应用,懒汉式含有一层懒加载(lazy loading)的概念,也叫做惰性初始化。

public class ElvisLazyLoading {
    private static ElvisLazyLoading instance;
    private ElvisLazyLoading(){}
    public static ElvisLazyLoading newInstance(){
        if(instance == null){
            instance = new ElvisLazyLoading();
        }
        return instance;
    }
}

初始的时候不会对INSTANCE进行初始化,它的默认值是null,在调用 newInstance 方法时会判断,若INSTANCE为null,则会把INSTANCE的引用指向ElvisLazyLoading的构造方法。


这种方式能够实现一个懒加载的思想,但是 这种写法会存在并发问题,由于多线程各自运行自己的执行路径,当同时执行到 INSTANCE = new ElvisLazyLoading() 代码时,各自的线程都认为自己应该创建一个新的ElvisLazyLoading对象,所以最后的结果可能会存在多个ElvisLazyLoading 实例,所以这种方式不推荐使用。


尝试加锁


很显然的,可以尝试对 newInstance() 方法加锁来避免产生并发问题,但是这种方式不可能,由 synchronized 加锁会导致整个方法开销太大,在遇见类似问题时,应该尝试换一种方式来解决,而不应该只通过简单粗暴的加锁来解决一切并发问题。

public synchronized static ElvisLazyLoading newInstance(){
  if(INSTANCE == null){
    INSTANCE = new ElvisLazyLoading();
  }
  return INSTANCE;
}

同步代码块


 synchronized 关键字不仅可以锁住方法的执行,也可以对方法中的某一块代码进行锁定,也叫做同步代码块

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

不要觉得只要加锁了,就不会存在线程安全问题,线程是Java中很重要的一个课题,需要细细研究。这种同步代码块的方式也会存在线程安全问题,当多个线程同时判断自己的singleton 实例为null的时候,同样会创建多个实例。


双重检查


Double-Check概念对于多线程开发者来说不会陌生,如代码中所示,我们进行了两次if (instance == null)检查,这样就可以保证线程安全了。这样,实例化代码只用执行一次,后面再次访问时,判断if (instance == null),直接return实例化对象。

public class ElvisDoubleCheck {
    private static volatile ElvisDoubleCheck instance;
    private ElvisDoubleCheck(){}
    public static ElvisDoubleCheck newInstance(){
        if(instance == null){
            synchronized (ElvisDoubleCheck.class){
                if (instance == null){
                    instance = new ElvisDoubleCheck();
                }
            }
        }
        return instance;
    }
}

优点:线程安全;延迟加载;效率较高。


静态内部类单例


静态内部类的单例与饿汉式采用的机制类似,但又有不同。两者都是采用了类装载的机制来保证初始化实例时只有一个线程。不同的地方在饿汉式方式是只要Elvis类被装载就会实例化,没有Lazy-Loading的作用,而静态内部类方式在ElvisStaticInnerClass类被装载时并不会立即实例化,而是在需要实例化时,调用newInstance方法,才会装载SingletonInstance类,从而完成ElvisStaticInnerClass的实例化。

public class ElvisStaticInnerClass {
    private ElvisStaticInnerClass(){}
    private static class SingletonInstance{
        private static final ElvisStaticInnerClass instance = new ElvisStaticInnerClass();
    }
    public static ElvisStaticInnerClass newInstance(){
        return SingletonInstance.instance;
    }
}

优点:避免了线程不安全,延迟加载,效率高。


枚举单例


实现Singleton的第四种方法是声明一个包含单个元素的枚举类型

public enum  ElvisEnum {
    INSTANCE;
    public void leaveTheBuilding(){}
}

这种方法在功能上与公有域方法相似,但更加简洁。无偿地提供了序列化机制,有效防止多次实例化,即使在面对复杂的序列化或者反射攻击的时候。单元素的枚举类型经常成为实现Singleton的最佳方法


优点: 系统内存中该类只存在一个对象,节省了系统资源,对于一些需要频繁创建销毁的对象,使用单例模式可以提高系统性能。


缺点:当想实例化一个单例类的时候,必须要记住使用相应的获取对象的方法,而不是使用new,可能会给其他开发人员造成困扰,特别是看不到源码的时候。


后记


看完本文,你是否对**构造器私有、枚举和单例**这个主题有了新的认知呢?


你至少应该了解:


1. 单例模式的几种写法及其优缺点分析

2. 为什么反射能够对私有构造器产生破坏?

3. 有哪几种比较好用的线程安全的单例模式?

相关文章
|
2月前
|
安全 Java 测试技术
🎉Java零基础:全面解析枚举的强大功能
【10月更文挑战第19天】本文收录于「滚雪球学Java」专栏,专业攻坚指数级提升,希望能够助你一臂之力,帮你早日登顶实现财富自由🚀;同时,欢迎大家关注&&收藏&&订阅!持续更新中,up!up!up!!
124 60
|
2月前
|
Java
java小工具util系列4:基础工具代码(Msg、PageResult、Response、常量、枚举)
java小工具util系列4:基础工具代码(Msg、PageResult、Response、常量、枚举)
56 24
|
4月前
|
安全 Java 索引
Java——反射&枚举
本文介绍了Java反射机制及其应用,包括获取Class对象、构造方法、成员变量和成员方法。反射允许在运行时动态操作类和对象,例如创建对象、调用方法和访问字段。文章详细解释了不同方法的使用方式及其注意事项,并展示了如何通过反射获取类的各种信息。此外,还介绍了枚举类型的特点和使用方法,包括枚举的构造方法及其在反射中的特殊处理。
81 9
Java——反射&枚举
|
4月前
|
Java
java小工具util系列4:基础工具代码(Msg、PageResult、Response、常量、枚举)
java小工具util系列4:基础工具代码(Msg、PageResult、Response、常量、枚举)
96 5
|
4月前
|
安全 Java 开发者
Java 枚举(enum)详解
Java 中的枚举(`enum`)是一种特殊的数据类型,用于定义一组固定的常量,提升代码的类型安全性和可读性。枚举使用 `enum` 关键字定义,支持方法和构造函数,具有类型安全、单例、自动序列化等特点,并且可以遍历和用于 `switch` 语句中。实际应用包括状态机、指令集、类型标识等场景。枚举使代码更加清晰易维护。
295 1
|
4月前
|
Oracle Java 关系型数据库
【颠覆性升级】JDK 22:超级构造器与区域锁,重塑Java编程的两大基石!
【9月更文挑战第6天】JDK 22的发布标志着Java编程语言在性能和灵活性方面迈出了重要的一步。超级构造器和区域锁这两大基石的引入,不仅简化了代码设计,提高了开发效率,还优化了垃圾收集器的性能,降低了应用延迟。这些改进不仅展示了Oracle在Java生态系统中的持续改进和创新精神,也为广大Java开发者提供了更多的可能性和便利。我们有理由相信,在未来的Java编程中,这些新特性将发挥越来越重要的作用,推动Java技术不断向前发展。
|
5月前
|
Java
Java枚举使用的基本案例
这篇文章是关于Java枚举的基本使用,通过一个指令下发的代码案例,展示了如何定义枚举、使用枚举以及如何通过枚举实现指令的匹配和处理。
|
5月前
|
Java 开发者
在Java编程中,if-else与switch作为核心的条件控制语句,各有千秋。if-else基于条件分支,适用于复杂逻辑;而switch则擅长处理枚举或固定选项列表,提供简洁高效的解决方案
在Java编程中,if-else与switch作为核心的条件控制语句,各有千秋。if-else基于条件分支,适用于复杂逻辑;而switch则擅长处理枚举或固定选项列表,提供简洁高效的解决方案。本文通过技术综述及示例代码,剖析两者在性能上的差异。if-else具有短路特性,但条件增多时JVM会优化提升性能;switch则利用跳转表机制,在处理大量固定选项时表现出色。通过实验对比可见,switch在重复case值处理上通常更快。尽管如此,选择时还需兼顾代码的可读性和维护性。理解这些细节有助于开发者编写出既高效又优雅的Java代码。
70 2
|
5月前
|
设计模式 人工智能 Java
Java 如何使用单例类
Java 如何使用单例类
20 1
|
5月前
|
安全 Java 编译器