你真的了解java单例模式了吗?

 更新时间:2019年06月18日 16:38:05   作者:阿豪聊干货  
这篇文章主要介绍了你真的了解java单例模式了吗?实际上单例模式有着好几个变种,并且多线程中涉及到线程安全问题,,需要的朋友可以参考下

一、背景

最近在学习设计模式,在看到单例模式的时候,我一开始以为直接很了解单例模式了,实现起来也很简单,但是实际上单例模式有着好几个变种,并且多线程中涉及到线程安全问题,那么本文我们就来好好聊聊单例模式,说一下经典三种实现方式:饿汉式、懒汉式、登记式。并且解决掉多线程中可能出现的线程安全问题。

二、基本概念

1.为什么要使用单例模式?

在我们日常的工作中,很多对象通常占用非常重要的系统资源,比如:IO处理,数据库操作等,那我们必须要限制这些对象只有且始终使用一个公用的实例,即单例。

2.单例模式的实现方式

构造函数私有化,防止其他类生成唯一公用实例外的实例。且单例类应该被定义为final,也就是说单例类不能被继承,因为如果允许继承那子类就都可以创建实例,违背了类唯一实例的初衷。

类中一个静态变量来保存单实例的引用。

一个共有的静态方法来获取单实例的引用。
3.单例模式的UML类图

4.单例模式的经典实现方式

  • 饿汉式:一开始就创建好实例,每次调用直接返回,经典的“拿空间换时间”。
  • 懒汉式:延迟加载,第一次调用的时候才加载,然后返回,以后的每次的调用就直接返回。经典“拿时间换空间”,多线程环境下要注意解决线程安全的问题。
  • 登记式:对一组单例模式进行的维护,主要是在数量上的扩展,通过线程安全的map把单例存进去,这样在调用时,先判断该单例是否已经创建,是的话直接返回,不是的话创建一个登记到map中,再返回。

三、饿汉式---代码实现

1.单例类

package com.hafiz.designPattern.singleton;
/**
* Desc: 单例模式-饿汉式
* Created by hafiz.zhang on 2017/9/26.
*/
public class Singleton1 {
// 创建全局静态变量,保证只有一个实例
private static volatile Singleton1 instance = new Singleton1();
private Singleton1() {
// 构造函数私有化
System.out.println("--调用饿汉式单例模式的构造函数--");
}
public static Singleton1 getInstance() {
System.out.println("--调用饿汉式单例模式的静态方法返回实例--");
return instance;
}
}

2.测试类

public class DesignPatternTest {
@Test
public void testSingleton1() {
System.out.println("-----------------测试饿汉式单例模式开始--------------");
Singleton1 instance1 = Singleton1.getInstance();
System.out.println("第二次获取实例");
Singleton1 instance2 = Singleton1.getInstance();
System.out.println("instance1和instance2是否为同一实例?" + (instance1 == instance2));
System.out.println("-----------------测试饿汉式单例模式结束--------------");
}
}

3.测试结果

四、懒汉式---代码实现

1.单例类

package com.hafiz.designPattern.singleton;
/**
* Desc:单例模式-懒汉式
* Created by hafiz.zhang on 2017/9/26.
*/
public class Singleton2 {
// 创建全局静态变量,保证只有一个实例
private static Singleton2 instance = null;
// 构造函数私有化
private Singleton2() {
System.out.println("--调用懒汉式单例模式的构造方法--");
}
public static Singleton2 getInstance() {
System.out.println("--调用懒汉式单例模式获取实例--");
if (instance == null) {
System.out.println("--懒汉式单例实例未创建,先创建再返回--");
instance = new Singleton2();
}
return instance;
}
}

2.测试类

public class DesignPatternTest {
@Test
public void testSingleton2() {
System.out.println("-----------------测试懒汉式单例模式开始--------------");
Singleton2 instance1 = Singleton2.getInstance();
System.out.println("第二次获取实例");
Singleton2 instance2 = Singleton2.getInstance();
System.out.println("instance1和instance2是否为同一实例?" + (instance1 == instance2));
System.out.println("-----------------测试懒汉式单例模式结束--------------");
}
}

3.测试结果

细心的同学已经发现,这种实现方式,在多线程的环境中,是有线程安全安全问题的,有可能两个或多个线程判断instance都为null,然后创建了好几遍实例,不符合单例的思想,我们可以对它进行改进。

五、改进懒汉式1---代码实现

原理:使用JDK的synchronized同步代码块来解决懒汉式线程安全问题。

1.单例类

package com.hafiz.designPattern.singleton;
/**
* Desc:单例模式-懒汉式
* Created by hafiz.zhang on 2017/9/26.
*/
public class Singleton2 {
// 创建全局静态变量,保证只有一个实例
private static Singleton2 instance = null;
// 构造函数私有化
private Singleton2() {
System.out.println("--调用懒汉式单例模式的构造方法--");
}

public static Singleton2 getInstance() {
System.out.println("--调用懒汉式单例模式获取实例--");
     if (instance != null) {
        System.out.println("--懒汉式单例实例已经创建,直接返回--");
 return instance;
     }
synchronized (Singleton2.class) {
   if (instance == null) {
  System.out.println("--懒汉式单例实例未创建,先创建再返回--");
  instance = new Singleton2();
   }
}
return instance;
}
} 

2.测试结果

六、改进懒汉式2---代码实现

原理:使用JVM隐含的同步和类级内部类来解决,JVM隐含的同步解决了多线程情况下线程安全的问题,类级内部类解决只有使用的时候才加载(延迟加载)的问题。

1.JVM隐含的同步有哪些?

静态初始化器(在静态字段上或static{}静态代码块的初始化器)初始化数据时

访问final字段时

在创建线程之前创建对象时

线程可以看见它将要处理的对象时

2.什么是类级内部类?

有static修饰的成员式内部类。没有static修饰的成员式内部类叫对象级内部类。

类级内部类相当于其外部类的static成分,他的对象与外部类对象间不存在依赖关系,因此可直接创建,而对象级内部类的实例,是绑定在外部对象实例中的。

类级内部类中,可以定义静态的方法。在静态的方法中只能够引用外部类的中的静态成员方法或者成员变量

类级内部类相当于其外部类的成员,只有在第一次被使用的时候才会被装载

3.单例类

package com.hafiz.designPattern.singleton;
/**
* Desc:单例模式-改进懒汉式
* Created by hafiz.zhang on 2017/9/26.
*/
public class Singleton3 {
private static class Singleton4 {
private static Singleton3 instance;
static {
System.out.println("--类级内部类被加载--");
instance = new Singleton3();
}
private Singleton4() {
System.out.println("--调用类级内部类的构造函数--");
}
}
private Singleton3() {
System.out.println("--调用构造函数--");
}
public static Singleton3 getInstance() {
System.out.println("--开始调用共有方法返回实例--");
Singleton3 instance;
System.out.println("---------------------------");
instance = Singleton4.instance;
System.out.println("返回单例");
return instance;
}
}

4.测试类

package com.hafiz.www;
import com.hafiz.designPattern.observer.ConcreteObserver;
import com.hafiz.designPattern.observer.ConcreteSubject;
import com.hafiz.designPattern.singleton.Singleton1;
import com.hafiz.designPattern.singleton.Singleton2;
import com.hafiz.designPattern.singleton.Singleton3;
import com.hafiz.designPattern.singleton.Singleton4;
import com.hafiz.designPattern.singleton.Singleton4Child1;
import com.hafiz.designPattern.singleton.SingletonChild2;
import org.junit.Test;
/**
* Desc:设计模式demo单元测试类
* Created by hafiz.zhang on 2017/7/27.
*/
public class DesignPatternTest {
@Test
public void testSingleton3() {
System.out.println("-----------------测试改进懒汉式单例模式开始--------------");
Singleton3 instance1 = Singleton3.getInstance();
System.out.println("第二次获取实例");
Singleton3 instance2 = Singleton3.getInstance();
System.out.println("instance1和instance2是否为同一实例?" + (instance1 == instance2));
System.out.println("-----------------测试改进懒汉式单例模式结束--------------");
}
}

5.测试结果

七、登记式--代码实现

1.基类

package com.hafiz.designPattern.singleton;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;
/**
* Desc: 单例模式-登记式
* Created by hafiz.zhang on 2017/9/26.
*/
public class Singleton4 {
private static Map<String, Singleton4> map = new ConcurrentHashMap<>();
protected Singleton4() {
System.out.println("--私有化构造函数被调用--");
}
public static Singleton4 getInstance(String name) {
if (name == null) {
name = Singleton4.class.getName();
System.out.println("--name为空,默认赋值为:--" + Singleton4.class.getName());
}
if (map.get(name) != null) {
System.out.println("name对应的值存在,直接返回");
return map.get(name);
}
System.out.println("name对应的值不存在,先创建,再返回");
try {
Singleton4 result = (Singleton4)Class.forName(name).newInstance();
map.put(name, result);
return result;
} catch (InstantiationException e) {
e.printStackTrace();
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (ClassNotFoundException e) {
e.printStackTrace();
}
return null;
}
public Map<String, Singleton4> getMap() {
return map;
}
}

2.子类1

package com.hafiz.designPattern.singleton;
/**
* Desc:
* Created by hafiz.zhang on 2017/9/26.
*/
public class Singleton4Child1 extends Singleton4 {

public static Singleton4Child1 getInstance() {
return (Singleton4Child1) Singleton4.getInstance("com.hafiz.designPattern.singleton.Singleton4Child1");
}
}

3.子类2

package com.hafiz.designPattern.singleton;
/**
* Desc:
* Created by hafiz.zhang on 2017/9/26.
*/
public class SingletonChild2 extends Singleton4 {
public static SingletonChild2 getInstance() {
return (SingletonChild2) Singleton4.getInstance("com.hafiz.designPattern.singleton.SingletonChild2");
}
}

4.测试类

public class DesignPatternTest {
@Test
public void testSingleton4() {
System.out.println("-----------------测试登记式单例模式开始--------------");
System.out.println("第一次取得实例");
Singleton4 instance1 = Singleton4.getInstance(null);
System.out.println("res:" + instance1);
System.out.println("第二次获取实例");
Singleton4Child1 instance2 = Singleton4Child1.getInstance();
System.out.println("res:" + instance2);
System.out.println("第三次获取实例");
SingletonChild2 instance3 = SingletonChild2.getInstance();
System.out.println("res:" + instance3);
System.out.println("第四次获取实例");
SingletonChild2 instance4 = new SingletonChild2();
System.out.println("res:" + instance4);
System.out.println("输出父类Map中所有的单例");
Map<String, Singleton4> map = instance1.getMap();
for (Map.Entry<String, Singleton4> item : map.entrySet()) {
System.out.println("map-item:" + item.getKey() + "=" + item.getValue());
}
System.out.println("instance1和instance2是否为同一实例?" + (instance1 == instance2));
System.out.println("-----------------测试登记式单例模式结束--------------");
}
}

5.测试结果

该解决方案的缺点:基类的构造函数对子类公开了(protected),有好的解决方案的博友可以讨论指教~

八、总结

经过本文,我们就搞明白了什么叫单例模式,如何优雅的实现经典的单例模式,如何进行拓展和开发具有线程安全的单例模式。对于我们以后的开发非常有帮助,也让我们更加了解单例模式。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • Java使用excel工具类导出对象功能示例

    Java使用excel工具类导出对象功能示例

    这篇文章主要介绍了Java使用excel工具类导出对象功能,结合实例形式分析了java创建及导出Excel数据的具体步骤与相关操作技巧,需要的朋友可以参考下
    2017-10-10
  • 浅谈一下Java线程组ThreadGroup

    浅谈一下Java线程组ThreadGroup

    ThreadGroup是为了方便线程管理出现了,可以统一设定线程组的一些属性,比如setDaemon,设置未处理异常的处理方法,设置统一的安全策略等等,需要的朋友可以参考下
    2023-05-05
  • java统计字符串单词个数的方法解析

    java统计字符串单词个数的方法解析

    在一些项目中可能需要对一段字符串中的单词进行统计,本文在这里分享了一个简单的demo,有需要的朋友可以拿去看一下
    2017-01-01
  • Java 面向对象和封装全面梳理总结

    Java 面向对象和封装全面梳理总结

    面向对象乃是Java语言的核心,是程序设计的思想,在面向对象程式设计方法中,封装(英语:Encapsulation)是指一种将抽象性函式接口的实现细节部分包装、隐藏起来的方法。封装可以被认为是一个保护屏障,防止该类的代码和数据被外部类定义的代码随机访问
    2021-10-10
  • java中如何获取时间戳的方法实例

    java中如何获取时间戳的方法实例

    时间戳通常是一个字符序列,唯一地标识某一刻的时间,所以下面这篇文章主要给大家介绍了关于java中如何获取时间戳的相关资料,文中通过示例代码介绍的非常详细,需要的朋友可以参考借鉴,下面随着小编来一起学习学习吧。
    2017-11-11
  • 使用@ApiModel遇到的问题及解决

    使用@ApiModel遇到的问题及解决

    这篇文章主要介绍了使用@ApiModel遇到的问题及解决,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-06-06
  • SpringBoot项目没有把依赖的jar包一起打包的问题解决

    SpringBoot项目没有把依赖的jar包一起打包的问题解决

    这篇文章主要介绍了SpringBoot项目没有把依赖的jar包一起打包的问题解决,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2019-09-09
  • SpringBoot+Netty实现简单聊天室的示例代码

    SpringBoot+Netty实现简单聊天室的示例代码

    这篇文章主要介绍了如何利用SpringBoot Netty实现简单聊天室,文中的示例代码讲解详细,对我们学习SpringBoot有一定帮助,感兴趣的同学可以了解一下
    2022-02-02
  • java连接Mongodb实现增删改查

    java连接Mongodb实现增删改查

    这篇文章主要为大家详细介绍了java连接Mongodb实现增删改查,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2020-03-03
  • java 两阶段终止线程的正确做法

    java 两阶段终止线程的正确做法

    这篇文章主要给大家分享了java 两阶段终止线程的正确做法,文章列举出错误的做法与正确做法做对比,具有一定的参考价值,需要的小伙伴可以参考一下,希望对你有所帮助
    2021-12-12

最新评论