面试官:写一个单例模式

1. 什么是单例模式

了解单例模式之前,我们需要先了解什么是设计模式。

设计模式是一种抽象的编程思想,不局限于编程语言,简单来说,就是一些大佬程序猿针对一些典型的场景,给出一些典型的解决方案,只要按照这个方案来,也就是按照给定的设计模式,此时代码写的差,也差不到哪去。

本节讲解的就是设计模式中的其中一种:单例模式

设计模式本质就是一些规章制度,就比如变量命名的风格,如果这个模块要求使用某某设计模式,可以不用吗?当然可以,大不了被队友喷呗,只要内心足够强大。

让我印象特别深的一件事,我还在上学的时候,班里有个人觉得自己写的代码很厉害,往班级群里发,我一看那代码,我靠,shujuyuan,lianbiao,kehuduanqingqiu,这都什么变量名啊,代码能跑吗?能跑,这如果跟我做同事,我指定得喷他。

单例模式很好理解,单例单例,只能有单个实例嘛,就是一个类只能有实例一个对象嘛,不要理解成这个类只能 new 一次哈,而是这个类压根就不让你 new,但是会提供一个给你获取对象的方法,每次获取都是相同的对象!

单例模式的实现常见的有两种实现方式,分别是饿汉模式,懒汉模式,名字奇怪不要紧,后续会讲解。


2. 饿汉模式的实现


public class SingleHungry {
    // 提前创建好对象
    private static final SingleHungry instance = new SingleHungry();

    // 对外隐藏构造方法
    private SingleHungry() {

    }

    // 对外提供获取这个对象的方法
    public static SingleHungry getInstance() {
        return instance;
    }
}

此时饿汉版本的单例模式就实现完成了,阅读上述代码,我们已经提前创建好了 SingleHungry 的实例对象 instance 了,而且使用 private 关键字修饰了构造方法,也就是对外不提供构造方法了,紧接着又写了一个获取 instance 的方法,这样一来,每次获取到的实例都是同一个实例!


public class ThreadDemo {
    public static void main(String[] args) {
        SingleHungry instance1 = SingleHungry.getInstance();
        SingleHungry instance2 = SingleHungry.getInstance();
        System.out.println(instance1.equals(instance2));
    }
}
// 打印结果:true

当然如果你尝试用 SingleHungry instance = new SingleHungry();

这样显然是会报错的。

为什么这种方式叫做饿汉呢?由于 SingleHungry 类里面的 instance 变量是被 static 修饰的,表示是类的属性,所有对象共享的,只存在一份,并且这个 instance 是在类加载的时候被创建的,类加载是比较靠前的阶段,给人的感觉就是很着急,饿得慌,所以就叫做饿汉模式。


3. 懒汉模式的实现

懒汉模式则于饿汉模式相反,饿汉不是很着急吗,老早就把对象创建了,而懒汉模式则是你什么时候需要,我再创建,那么很多小伙伴想到了,这不是很简单吗?直接在 getInstance() 加一个判断嘛,具体代码实现如下:


public class SingleLazy {
    // 不提前创建好对象
    private static SingleLazy instance = null;

    // 对外隐藏构造方法
    private SingleLazy() {

    }

    // 对外提供获取这个对象的方法
    public static SingleLazy getSingleLazy() {
        if (instance == null) {
            instance = new SingleLazy();
        }
        return instance;
    }
}

那么对于懒汉模式来说,什么时候调用 SingleLazy.getSingleLazy() 的时候才会创建 SingleLazy 对象,效率显然是比饿汉模式要高的。


4. 多线程使用单例模式

大家可千万不要忘了,咱们的标题可还是多线程呐,如果单例模式仅仅是在单线程,那执行顺序都是唯一的,显然不会出现线程安全的问题,但是由于多线程环境下的随机调度,抢占式执行,这样一来,可能就会出问题。

此时来判断上述饿汉模式和懒汉模式在多线程环境下是否会出现线程安全的问题呢?

这里我们主要来看这两个版本里最主要的 get 方法的实现:


// 饿汉模式
public static SingleHungry getInstance() {
    return instance;
}

// 懒汉模式
public static SingleLazy getInstance() {
    if (instance == null) {
        instance = new SingleLazy();
    }
    return instance;
}

此时很明显发现,饿汉模式的实现,只要调用 get 方法,就会直接返回,不涉及对 instance 的修改,只涉及读操作,而懒汉模式中涉及到 load,cmp,new,save等,简直就是又有读和写啊。

前面讲线程安全时,涉及到读和写,都有可能出现线程不安全。

就懒汉模式的 getInstance 方法,可以大致分为四个步骤,先读取 instance 的值(load),于 null 做比较(cmp) 条件满足,进行 new 操作(new 也分为好几个步骤),接着在写回 instance(save)。

下面我们就画图模拟两个线程里都调用懒汉模式 getInstance 方法的情况:

上述情况之所以能触发多次 new 操作,本质上是因为比较,读,写这三个操作不是原子的,于是我们就可以通过加锁针对上述 getSingleLazy 方法做优化,保证原子性。


// 懒汉模式
public static SingleLazy getInstance() {
    synchronized (SingleLazy.class) {
        if (instance == null) {
            instance = new SingleLazy();
        }
    }
    return instance;
}

此时我们是解决了原子性的问题,但是这个代码会不会效率很低呢?

每次调用 getSingleLazy 方法的时候,都需要先加锁!加锁操作也是有开销的,那我们有必要每次调用该方法都加锁吗?

其实不用每次都加锁,当 instance 为 null 的时候,需要 new 对象的时候,才需要加锁,只要第一次 new 过对象了,后续都不需要加锁了,因为这个类只能实例一个对象,于是我们就可以在前面再次加上判断语句:


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

写到这里,我们还得思考,这个代码是否有内存可见性的问题,假设同一时间很多线程里面都调用了 getSingleLazy 方法,此时只能有一个线程在进行 new 操作,其他线程都在等待锁释放,那么那么多线程读 instance 发现都是 null。

当锁释放,另一个线程获取到锁,前面发现第一次 if 里面读的 instance 为 null,进入第二个 if 的时候,就没有从内存中再次读 instance,而是直接去 寄存器/cache 里读,所以仍然判断 instance 为 null,也会出现重复 new 对象的操作。

再者,instance = new SingleLazy(); 这个操作可以大致拆分成三个步骤:

1. 申请内存空间
2. 调用对应构造方法,初始化内存空间
3. 把内存空间的地址赋值给 instance 引用

如果编译器优化,指令重排序了,本来是按照 1 2 3 的顺序,如果优化成了 1 3 2 的顺序,在多线程的情况,1 2 3 顺序和 1 3 2 是一样的结果,但是多线程环境可不一定了!

假设 t1 线程按照 1 3 2 的顺序去执行,执行完 3 这个操作后,此时 instance 里只是存了一个地址,但地址对应的内存空间并没有初始化,此时如果 t1 被 CPU 切走了(CPU调用其他线程了),CPU 开始调用 t2 线程执行 1 2 3 顺序,此时 t2 就发现 instance 里不为 null,直接 return instance; 但是这个 instance 对应的内存空间并未初始化,所以 t2 调用 getInstance() 得到的就是一个非法的对象(未初始化的对象)。

如上可知,我们要避免出现内存可见性和指令重排序的问题,这时就可以用我们前面学到过的 volatile 关键字,来保证内存可见性,禁止指令重排序。

最终完整版的懒汉模式代码如下:


public class SingleLazy {
    // 不提前创建好对象
    private volatile static SingleLazy instance = null;

    // 对外隐藏构造方法
    private SingleLazy() {

    }

    // 对外提供获取这个对象的方法
    public static SingleLazy getSingleLazy() {
        if (instance == null) {
            synchronized (SingleLazy.class) {
                if (instance == null) {
                    instance = new SingleLazy();
                }
            }
        }
        return instance;
    }
}

当然这个代码还不完整!虽然我们构造方法是私有的,但是利用反射仍然可以构造多个对象,这里可以采用枚举来防止被反射,还有防止序列化的情况,但是反射是非常规编程手段,如果被面试官要求写个懒汉模式,其实写到这,已经够了!


下期预告:【多线程】生产者消费者模型

  • 22
    点赞
  • 22
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 19
    评论
当描述Java面试者的专业技能时,可以包括以下方面: 1. 编程语言和框架:列出熟悉的Java编程语言和相关框架,例如Spring、Hibernate、Struts等。 示例:熟练掌握Java编程语言,对Spring和Hibernate框架有深入理解并具有实际项目经验。 2. 数据库和SQL:描述熟悉的数据库系统和SQL语言能力,如MySQL、Oracle、SQL Server等。 示例:熟练运用MySQL数据库,并熟悉SQL语言进行数据操作和优化。 3. Web开发:包括前端和后端开发技能,如HTML、CSS、JavaScript、Servlet等。 示例:具备基本的前端开发技能,熟悉HTML、CSS和JavaScript,并能使用Servlet开发Java Web应用。 4. 设计模式:熟悉常用的设计模式,如单例模式、工厂模式、观察者模式等。 示例:熟悉常用的设计模式,并能在项目中恰当地应用以提高代码可维护性和扩展性。 5. 版本控制工具:描述使用过的版本控制工具,如Git、SVN等。 示例:熟练使用Git进行团队协作和代码版本控制。 6. 测试和调试:描述熟悉的测试工具和调试技能,如JUnit、Mockito、Eclipse调试器等。 示例:熟练使用JUnit进行单元测试,并能运用Mockito进行模拟测试。 7. 性能优化:描述对Java应用程序进行性能优化的能力,如代码优化、数据库查询优化等。 示例:具备对Java应用程序进行性能调优的实践经验,包括代码优化和数据库查询优化。 除了上述示例,你可以根据自己的实际经验和技能做适当调整和补充。记住要准确、简明地描述你的技能,以便面试官能够快速了解你的专业能力。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 19
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

程序猿教你打篮球

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值