多线程系列七之单例模式
提示:本文主要讲述线程中如何实现单例模式和工厂模式
文章目录
一、单例模式
什么是单例模式?
单例模式是一种设计模式。在一些特殊场景中,某些特定的类只能创建一个实例,不应该实例化多个对象。
1.注意:像这种业务需求(jdbc、DataSource),我们也不是必须使用单例模式,也可以约定成一种君子协定,但这样就加大犯错的机会。而我们使用单例模式后,想创建多个实例都没有办法创建
2.在单例模式中我们将主要介绍两种模式,分别是饿汉模式和懒汉模式
1) 饿汉模式
1.什么是饿汉模式?
一、这种设计模式之所以被成为饿汉模式,是因为类加载阶段就把这个实例创建出来了,因此称之为饿汉模式。
1.这个属性instance被static修饰,因此这个属性与实例无关,而是与类相关的。
2.Java中的代码,每个类在编译完成之后都会形成一个 .class文件。在JVM运行过程中,就会加载这个.class文件,并读取其中的二进制指令,在内存中构造出对应的类对象。例如:Singleton.class
3.一个类对象在一个Java进程中只能有一份,因此类对象内部的类属性也是唯一一份的,这个类属性被赋值了new Singleton()因此这个实例也是唯一一份的。
2.代码演示
3.代码分析
1.这里输出的结果是true,说明赋值给s1 和 s2的是在类加载阶段创建的唯一一个实例。
2.类加载阶段就把实例创建出来了,类加载是比较靠前的阶段,给人一种比较急切的感觉,所以我们称之为饿汉模式。
1、什么是类加载???
运行一个Java程序,我们就需要让Java进程找到对应的.class文件,读取文件内容,解析,构造类对象等这一系列的操作我们就称之为类类加载。
2、我们这种设计是如何保证单例呢???
1.成员属性被static修饰,使我们让这个instance这个属性变成了类属性。类属性是长在类对象上的,而类对象在一个进程中是唯一一份的。类加载只进行一次,在类加载阶段类对象被创建。
2.构造方法被private修饰,我们就无法在类外部通过new关键字创建出来实例。
4.类属性和普通属性的区别
后续代码无论如何修改在内存中都只有一份,A.class都只有一份,而属性y被static修饰成为类属性也落到了类对象的空间内,因此无论如何修改,都是在相同的一块空间对数值进行修改。
2) 懒汉模式
1.什么是懒汉模式
懒汉模式是指唯一实例没有随着类加载而创建,而是在外部类第一次使用这个实例时候才创建。如果一直不使用,那就不去创建,比较慵懒,所以称之为懒汉模式、
2.代码演示
运行结果:
二、饿汉模式代码改进
1) 两种模式哪种存在线程安全问题
思考: 在这两种单例模式中是否存在线程安全问题呢???
答案:
这里主要从两种模式的get方法入手,因为产生线程安全问题因为代码操作不是原子的,因为操作系统的随机调度,抢占式执行,容易造成脏读问题。
由图可见,饿汉模式中只设计读操作,没有线程安全问题,而懒汉模式中既有读操作又有写操作,操作并不是原子的,可能会因为线程的随机调度,抢占式执行影响线程安全。
2) 分析解决
由图可知我们需要使用synchronized进行加锁,是操作是原子的。
这样就可以保证t2不会读到t1的脏数据!!!
1.但是这样如果我们使用这个唯一实例十分频繁,那么每次都要进行加锁操作嘛?其实只有第一次创建和第二次容易产生脏读问题,在之后的使用这个实例if条件不满足直接返回instance了,
2.代码还能改进嘛???
每次加锁释放锁也需要消耗资源,需要开销的。但是实例创建后 后续其实不再需要
3.因此我们把代码还可以做如下改进!!!
注意:
1.此时懒汉模式的线程安全还没有完全解决
2.因为new操作本身不是原子的,可能会涉及指令重排序的问题
3.指令重排序问题将在线程安全系列以例子进行讲解改进