Java面试必备:手写单例模式

面试官:请手写下几种常见的单例模式
我:好的(面带微笑),心里暗喜(送分题)。
没成想提笔便写出了如此豪放的代码,不堪回首,请原谅我的不羁!

此篇整理了几种常见的单例模式代码示例,再有面试官让手撕单例模式,便能心中有码,下笔有神。

为什么要有单例模式

实际编程应用场景中,有一些对象其实我们只需要一个,比如线程池对象、缓存、系统全局配置对象等。这样可以就保证一个在全局使用的类不被频繁地创建与销毁,节省系统资源。

实现单例模式的几个要点

  1. 首先要确保全局只有一个类的实例。
    要保证这一点,至少类的构造器要私有化。
  2. 单例的类只能自己创建自己的实例。
    因为,构造器私有了,但是还要有一个实例,只能自己创建咯!
  3. 单例类必须能够提供自己的唯一实例给其他类
    就是要有一个公共的方法能返回该单例类的唯一实例。

    单例模式的6种实现

    1、饿汉式—静态常量方式(线程安全)

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

类加载时就初始化实例,避免了多线程同步问题。天然线程安全。

2、饿汉式—静态代码块方式(线程安全)

其实就是在上面 静态常量饿汉式 实现上稍微变动了一下,将类的实例化放在了静态代码块中而已。其他没区别。

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

3、懒汉式(线程不安全)

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

这是最基本的实现方式,第一次调用才初始化,实现了懒加载的特性。多线程场景下禁止使用,因为可能会产生多个对象,不再是单例。

4、懒汉式(线程安全,方法上加同步锁)

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

和上面 懒汉式(线程不安全)实现上唯一不同是:获取实例的getInstance()方法上加了同步锁。保证了多线程场景下的单例。但是效率会有所折损,不过还好。

5、双重校验锁(线程安全,效率高)

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

此种实现中不用每次需要获得锁,减少了获取锁和等待的事件。
注意volatile关键字的使用,保证了各线程对singleton静态实例域修改的可见性。

6、静态内部类实现单例(线程安全、效率高)

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

这种方式下 Singleton 类被装载了,instance 不一定被初始化。因为 SingletonHolder 类没有被主动使用,只有通过显式调用 getInstance 方法时,才会显式装载 SingletonHolder 类,从而实例化 instance。
注意内部类SingletonHolder要用static修饰且其中的静态变量INSTANCE必须是final的。

此篇完。单例模式掌握至此,足以应付“手写单例”的面试场景。

扫码关注微信公众号:二营长的笔记。回复“二营长”,可领取Java相关技术资料。

原文地址:https://www.cnblogs.com/happyone/p/11221157.html

时间: 2024-10-06 13:31:15

Java面试必备:手写单例模式的相关文章

java面试之手撕代码

在最近的面试中,面试官经常要面试者手撕代码,哎奈何本人只知道原理,一到手撕就死乔乔了. 第一题:宜信一面手撕代码之写代码实现一个栈的的前序遍历: 第一步.要构造节点类 public class Node { private int data; private Node leftChild;//左孩子 private Node rightChild;//右孩子 public Node(int data,Node leftChild,Node rightChild){ this.data = dat

java架构之路(多线程)大厂方式手写单例模式

上期回顾: 上次博客我们说了我们的volatile关键字,我们知道volatile可以保证我们变量被修改马上刷回主存,并且可以有效的防止指令重排序,思想就是加了我们的内存屏障,再后面的多线程博客里还有说到很多的屏障问题. volatile虽然好用,但是别用的太多,咱们就这样想啊,一个被volatile修饰的变量持续性的在修改,每次修改都要及时的刷回主内存,我们讲JMM时,我们的CPU和主内存之间是通过总线来连接的,也就是说,每次我们的volatile变量改变了以后都需要经过总线,“道路就那么宽,

Java面试必备

1.面向对象的特征有哪些方面? 答:面向对象的特征主要有以下几个方面: - 抽象:抽象是将一类对象的共同特征总结出来构造类的过程,包括数据抽象和行为抽象两方面.抽象只关注对象有哪些属性和行为,并不关注这些行为的细节是什么. - 继承:继承是从已有类得到继承信息创建新类的过程.提供继承信息的类被称为父类(超类.基类):得到继承信息的类被称为子类(派生类).继承让变化中的软件系统有了一定的延续性,同时继承也是封装程序中可变因素的重要手段(如果不能理解请阅读阎宏博士的<Java与模式>或<设计

JAVA面试必备的知识宝典(一)

相关概念 面向对象的三个特征 封装,继承,多态.这个应该是人人皆知.有时候也会加上抽象. 多态的好处 允许不同类对象对同一消息做出响应,即同一消息可以根据发送对象的不同而采用多种不同的行为方式(发送消息就是函数调用).主要有以下优点: 可替换性:多态对已存在代码具有可替换性. 可扩充性:增加新的子类不影响已经存在的类结构. 接口性:多态是超累通过方法签名,想子类提供一个公共接口,由子类来完善或者重写它来实现的. 灵活性: 简化性: 代码中如何实现多态 实现多态主要有以下三种方式: 1. 接口实现

JAVA面试必备的知识宝典(二)

WeakReference与SoftReference的区别? 这点在四种引用类型中已经做了解释,这里简单说明一下即可: 虽然 WeakReference 与 SoftReference 都有利于提高 GC 和 内存的效率,但是 WeakReference ,一旦失去最后一个强引用,就会被 GC 回收,而软引用虽然不能阻止被回收,但是可以延迟到 JVM 内存不足的时候. 为什么要有不同的引用类型 不像C语言,我们可以控制内存的申请和释放,在Java中有时候我们需要适当的控制对象被回收的时机,因此

手写单例模式

/** * 单例模式 * @author gy2018 * */public class SingletonDemo {    /**     * 所谓的单例模式,就是整个程序有且仅有一个实例.     * 该类负责创建自己的对象 并确保只有一个对象被创建.     * 在java中,一般常用在工具类的实现或创建对象 需要消耗资源     *      * 特点:     * 类构造器私有     * 持有自己类型的属性     * 对外提供获取实例的静态方法     */          

JAVA面试必备的知识宝典(三)

数据类型相关 java中int char,long各占多少字节? |类型|位数|字节数| |-|-|-| |short|2|16| |int|4|32| |long|8|64| |float|4|32 |double|8|64| |char|2|16| 64位的JVM当中,int的长度是多少? Java 中,int 类型变量的长度是一个固定值,与平台无关,都是 32 位.意思就是说,在 32 位 和 64 位 的Java 虚拟机中,int 类型的长度是相同的. java int和Integer的

Java面试必备Springioc上

配置文件中 Proprety name值必须和 类中的成员变量private IUsedao  userDao一一对应 工程项目的代码为:

毕业两年的java面试总结及个人知识总结

前言:毕业两年多,一直从事java web方面的开发,感觉自己算踏入了一点点门槛吧,来记录下自己的成长吧,希望给广大网友们一些参考,共同进步.以下问题部分是面试被问到的,一些是自己的学习总结掌握了的知识,并没有被问到过,但仍需了解,不能为了面试而面试,这些知识仍需了解学习. 1. LinkedList 和 Arraylist区别 ? LinkedList可否模拟堆栈或者队列? 2. HashMap 和 HashTable区别? 3. 说一下HashMap 的put过程? key值怎么计算的?怎样