单例模式 分析代码优化方法
作者:Cream.icend 时间:2021-07-28 15:49:51
单例模式是23种设计模式之一,是比较简单的一种设计模式,它的目的是无论调用多少次,都返回同一个对象,它的特点是构造器私有化。
它分为两种结构,一种是懒汉式的,一种是饿汉式的,它们各有优缺点,我们先从饿汉式看起,代码如下:
public class Single {
private static Single single = new Single();
private Single() {
}
public Single getInstance() {
return single;
}
}
通过上面的程序可以看出来虽然我们加载同一个对象的目的确实达到了,但当程序被加载的时候就会创建single这个对象,当这个类有多个这样的方法时,我们可能会用不到这个对象中大多数单例,就会造成对内存的浪费。所以就出现了懒汉式的单例模式,代码如下:
public class Single {
private static Single single = null;
private Single() {
}
public Single getInstance() {
if(single==null){
single = new Single();
}
return single;
}
}
这样,只有当我们真正调用这个对象时它才会被new出来,但是这样是存在问题的。
当上面的第二段代码在第一次加载的时候有两个线程对其进行了调用,则会产生两个不同的对象,所以是线程不安全的,这时候就会想到给这个方法加个锁,加锁之后的代码如下:
public class Single {
private static Single single = null;
private Single() {
}
public synchronized Single getInstance() {
if (single == null) {
single = new Single();
}
return single;
}
}
这样做确实做到了线程安全,但是当加锁这个方法里面要执行很多东西,调用这个方法花费的时间会很长,这样对服务器来说是致命的,因为这个方法如果某个线程一直调用的话,其它的线程是没有办法调的,服务器就阻塞了,那么升级后的代码如下:
public class Single {
priate static Single single = null;
private Single() {
}
public Single getInstance() {
if (single == null) {
synchronized (Single.class) {
single = new Single();
}
}
return single;
}
}
仔细观察以后发现这样并没有锁住,当第一次同时有两个线程到达getInstance()方法if判断时,其中有一个肯定是阻塞的,当另外一个执行完以后,阻塞这个线程是不会再判断是否为空的,还是会创建一个对象的,这样又有多个对象被产生了,再对其进行升级,得到的代码如下:
public class Single {
private static Single single = null;
private Single() {
}
public Single getInstance() {
if (single == null) {
synchronized (Single.class) {
if (single == null) {
single = new Single();
}
}
}
return single;
}
}
这样就不会产生上面的问题,而且也只锁一次,因为第二次再执行这个方法时,会跳过if判断,直接返回single,不会再被锁,执行效率也会很高。
但即使是这样,也还是有问题的,因为我们不能确定在内存中是先给对象赋值,还是先创建了这个对象,所以第二个程序有可能得到的是初始化一半了的对象,在jdk1.5之后,我们可以用volatile这个关键字来避免这种情况,代码如下:
public class Single {
private static volatile Single single = null;
private Single() {
}
public Single getInstance() {
if (single == null) {
synchronized (Single.class) {
if (single == null) {
single = new Single();
}
}
}
return single;
}
}
但是这种情况很少使用,我在这里只是为了学习一下,嘻嘻
![](/images/zang.png)
![](/images/jiucuo.png)
猜你喜欢
SpringBoot使用protobuf格式的接口方式
![](https://img.aspxhome.com/file/2023/0/70950_0s.png)
java 算法之希尔排序详解及实现代码
Springmvc Controller接口代码示例
android6.0权限动态申请框架permissiondispatcher的方法
C#推送信息到APNs的方法
C#如何通过匿名类直接使用访问JSON数据详解
![](https://img.aspxhome.com/file/2023/7/67057_0s.jpg)
分析HashMap 的 JDK 源码
![](https://img.aspxhome.com/file/2023/4/68164_0s.jpg)
详解java Collections.sort的两种用法
![](https://img.aspxhome.com/file/2023/3/60473_0s.png)
在Java中按值调用和按引用调用
Java 是如何利用接口避免函数回调的方法
Spring实战之ResourceLoader接口资源加载用法示例
springboot中使用redis并且执行调试lua脚本
![](https://img.aspxhome.com/file/2023/1/61031_0s.png)
Java实现多用户注册登录的幸运抽奖
![](https://img.aspxhome.com/file/2023/5/57815_0s.png)
Android Flutter使用本地数据库编写备忘录应用
![](https://img.aspxhome.com/file/2023/6/109226_0s.webp)
如何使用Spring AOP的通知类型及创建通知
![](https://img.aspxhome.com/file/2023/2/61242_0s.png)
Java实现身份证号码验证源码示例分享
MyBatis-Plus拦截器实现数据权限控制的示例
Java面试题冲刺第二十五天--并发编程3
![](https://img.aspxhome.com/file/2023/9/58299_0s.png)
Springboot如何根据实体类生成数据库表
![](https://img.aspxhome.com/file/2023/7/59677_0s.jpg)