单利模式

单例模式(Singleton Pattern)是 Java 中最简单的设计模式之一。这种类型的设计模式属于创建型模式,它提供了一种创建对象的最佳方式。

这种模式涉及到一个单一的类,该类负责创建自己的对象,同时确保只有单个对象被创建。这个类提供了一种访问其唯一的对象的方式,可以直接访问,不需要实例化该类的对象。

1. 单例模式的结构

单例模式的主要有以下角色:

  • 单例类。只能创建一个实例的类
  • 访问类。使用单例类

2. 单例模式的实现

单例设计模式分两种:

饿汉式:类加载就会导致改单实例对象被创建

懒汉式:类加载不会导致改单实例对象被创建,而是首次使用该对象时才会创建

1.饿汉式-方式1(静态变量方式)

package cn.school.singleton;

/*
* 饿汉式
* 静态变量创建类的对象
* */ public class SingletonDemo1 {
//私有构造方法
private SingletonDemo1(){} //在成员位置创建该类的对象
private static SingletonDemo1 instance =new SingletonDemo1(); //对外提供静态方法获取类对象
public static SingletonDemo1 getInstance(){
return instance;//将创建好的对象直接返回,每次都是instance被返回。
}
}
package cn.school.singleton;

public class SingletonDemoMain {
public static void main(String[] args) {
// 创建两个对象进行测试
SingletonDemo1 obj1= SingletonDemo1.getInstance();
SingletonDemo1 obj2= SingletonDemo1.getInstance();
System.out.println(obj1);
System.out.println(obj2);
System.out.println(obj1==obj2);
}
}

说明

该方式在成员位置声明Singleton类型的静态变量,并创建Singleton类的对象instance。instance对象是随着类的加载而创建的。如果该对象足够大的话,而一直没有使用就会造成内存的浪费

2.饿汉式-方式2(静态代码块方式)

package cn.school.singleton;

public class SingletonDeom2 {
private SingletonDeom2(){}
private static SingletonDeom2 instance;
static {
instance =new SingletonDeom2();//在静态代码块中实例化对象。
}
//对外提供访问的方法
public static SingletonDeom2 getInstance(){
return instance;
}
}

说明:该方式在成员位置声明Singleton类型的静态变量,而对象的创建是在静态代码块中,也是对着类的加载而创建。所以和饿汉式的方式1基本上一样,当然该方式也存在内存浪费问题

3.懒汉式-方式1(线程不安全)

package cn.school.singleton;

/*
* 懒汉式
* 线程不安全
* */
public class SingletonDemo3 { //私有构造方法
private SingletonDemo3(){}
//在成员位置创建该类的对象
private static SingletonDemo3 instance; //对外提供静态方法获取该对象。
public static SingletonDemo3 getInstance(){
if(instance==null){
instance=new SingletonDemo3();
}
return instance;
}
}

说明

从上面代码我们可以看出该方式在成员位置声明Singleton类型的静态变量,并没有进行对象的赋值操作,当调用getInstance()方法获取Singleton类的对象的时候才创建Singleton类的对象,这样就实现了懒加载的效果。但是,如果是多线程环境,会出现线程安全问题。

4.懒汉式-方式2(线程安全)

package cn.school.singleton;

/*
* 懒汉式
* 线程安全
* */
public class SingletonDemo4 {
//私有构造方法
private SingletonDemo4(){} //在成员位置创建改类的对象
private static SingletonDemo4 instance;
//对外提供静态方法获取改对象,使用同步约束以下代码必须同步执行。确保多线程只创建一个对象
public static synchronized SingletonDemo4 getInstance(){
if(instance==null){
instance=new SingletonDemo4();
}
return instance;
}
}

说明:该方式也实现了懒加载效果,同时又解决了线程安全问题。但是在getInstance()方法上添加了synchronized关键字,导致该方法的执行效果特别低。从上面代码我们可以看出,其实就是在初始化instance的时候才会出现线程安全问题,一旦初始化完成就不存在了。

5.懒汉式-方式3(双重检查锁)

再来讨论一下懒汉模式中加锁的问题,对于 getInstance() 方法来说,绝大部分的操作都是读操作,读操作是线程安全的,所以我们没必让每个线程必须持有锁才能调用该方法,我们需要调整加锁的时机。由此也产生了一种新的实现模式:双重检查锁模式

双重检查锁模式是一种非常好的单例实现模式,解决了单例、性能、线程安全问题,上面的双重检测锁模式看上去完美无缺,其实是存在问题,在多线程的情况下,可能会出现空指针问题,出现问题的原因是JVM在实例化对象的时候会进行优化和指令重排序操作。

要解决双重检查锁模式带来空指针异常的问题,只需要使用 volatile 关键字, volatile 关键字可以保证可见性和有序性。

package cn.school.singleton;

public class SingletonDemo5 {
private SingletonDemo5(){} //定义静态成员变量,因为静态方法要用,所以变量必须是静态的
private static SingletonDemo5 instance; //对外提供静态方法获取该对象
public static SingletonDemo5 getInstance(){
//第一次判断,如果instance不为null,不进入抢锁阶段,直接返回实际
if(instance==null){
synchronized (SingletonDemo5.class){
//抢到锁之后再次判断是否为空
if (instance==null){
instance = new SingletonDemo5();
}
}
}
return instance;
}
}

小结:添加 volatile 关键字之后的双重检查锁模式是一种比较好的单例实现模式,能够保证在多线程的情况下线程安全也不会有性能问题。

6.懒汉式-方式4(静态内部类方式)

静态内部类单例模式中实例由内部类创建,由于 JVM 在加载外部类的过程中, 是不会加载静态内部类的, 只有内部类的属性/方法被调用时才会被加载, 并初始化其静态属性。静态属性由于被 static 修饰,保证只被实例化一次,并且严格保证实例化顺序。

package cn.school.singleton;

public class SingletonDemo6 {
private SingletonDemo6(){} //创建静态内部类
private static class SingletonHolder{
//在内部类创建常量对象。
private static final SingletonDemo6 INTANCE =new SingletonDemo6();
} //对外提供静态方法获取
public static SingletonDemo6 getInstance(){
return SingletonHolder.INTANCE;
}
}

说明

  • 第一次加载Singleton类时不会去初始化INSTANCE,只有第一次调用getInstance,虚拟机加载SingletonHolder并初始化INSTANCE,这样不仅能确保线程安全,也能保证 Singleton 类的唯一性。

小结

  • 静态内部类单例模式是一种优秀的单例模式,是开源项目中比较常用的一种单例模式。在没有加任何锁的情况下,保证了多线程下的安全,并且没有任何性能影响和空间的浪费。

7.枚举方式(饿汉式)

枚举类实现单例模式是极力推荐的单例实现模式,因为枚举类型是线程安全的,并且只会装载一次,设计者充分的利用了枚举的这个特性来实现单例模式,枚举的写法非常简单,而且枚举类型是所用单例实现中唯一一种不会被破坏的单例实现模式。

/**
* 枚举方式
*/
public enum Singleton {
INSTANCE;
}

3. 存在的问题

1.问题演示

破坏单例模式:

使上面定义的单例类(Singleton)可以创建多个对象,枚举方式除外。有两种方式,分别是序列化和反射。

  • 序列化反序列化

    Singleton类:

    public class Singleton implements Serializable {
    //私有构造方法
    private Singleton() {} private static class SingletonHolder {
    private static final Singleton INSTANCE = new Singleton();
    }
    //对外提供静态方法获取该对象
    public static Singleton getInstance() {
    return SingletonHolder.INSTANCE;
    }
    }

    Test类

    public class Test {
    public static void main(String[] args) throws Exception {
    //往文件中写对象
    //writeObject2File();
    //从文件中读取对象
    Singleton s1 = readObjectFromFile();
    Singleton s2 = readObjectFromFile(); //判断两个反序列化后的对象是否是同一个对象
    System.out.println(s1 == s2);
    } private static Singleton readObjectFromFile() throws Exception {
    //创建对象输入流对象
    ObjectInputStream ois = new ObjectInputStream(new FileInputStream("C:\\Users\\Think\\Desktop\\a.txt"));
    //第一个读取Singleton对象
    Singleton instance = (Singleton) ois.readObject(); return instance;
    } public static void writeObject2File() throws Exception {
    //获取Singleton类的对象
    Singleton instance = Singleton.getInstance();
    //创建对象输出流
    ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("C:\\Users\\Think\\Desktop\\a.txt"));
    //将instance对象写出到文件中
    oos.writeObject(instance);
    }
    }

    上面代码运行结果是false,表明序列化和反序列化已经破坏了单例设计模式。

  • 反射

    Singleton类

    public class Singleton {
    
        //私有构造方法
    private Singleton() {} private static volatile Singleton instance; //对外提供静态方法获取该对象
    public static Singleton getInstance() { if(instance != null) {
    return instance;
    }
    synchronized (Singleton.class) {
    if(instance != null) {
    return instance;
    }
    instance = new Singleton();
    return instance;
    }
    }
    }

    Test类

    public class Test {
    public static void main(String[] args) throws Exception {
    //获取Singleton类的字节码对象
    Class clazz = Singleton.class;
    //获取Singleton类的私有无参构造方法对象
    Constructor constructor = clazz.getDeclaredConstructor();
    //取消访问检查
    constructor.setAccessible(true); //创建Singleton类的对象s1
    Singleton s1 = (Singleton) constructor.newInstance();
    //创建Singleton类的对象s2
    Singleton s2 = (Singleton) constructor.newInstance(); //判断通过反射创建的两个Singleton对象是否是同一个对象
    System.out.println(s1 == s2);
    }
    }

    上面代码运行结果是false,表明序列化和反序列化已经破坏了单例设计模式

    注意:枚举方式不会出现这两个问题。

2.问题解决

  • 序列化、反序列方式破坏单例模式的解决方法

    在Singleton类中添加readResolve()方法,在反序列化时被反射调用,如果定义了这个方法,就返回这个方法的值,如果没有定义,则返回新new出来的对象。

    Singleton类:

    public class Singleton implements Serializable {
    
        //私有构造方法
    private Singleton() {} private static class SingletonHolder {
    private static final Singleton INSTANCE = new Singleton();
    } //对外提供静态方法获取该对象
    public static Singleton getInstance() {
    return SingletonHolder.INSTANCE;
    } /**
    * 下面是为了解决序列化反序列化破解单例模式
    */
    private Object readResolve() {
    return SingletonHolder.INSTANCE;
    }
    }

    源码解析:

    ObjectInputStream 类

    public final Object readObject() throws IOException, ClassNotFoundException{
    ...
    // if nested read, passHandle contains handle of enclosing object
    int outerHandle = passHandle;
    try {
    Object obj = readObject0(false);//重点查看readObject0方法
    .....
    } private Object readObject0(boolean unshared) throws IOException {
    ...
    try {
    switch (tc) {
    ...
    case TC_OBJECT:
    return checkResolve(readOrdinaryObject(unshared));//重点查看readOrdinaryObject方法
    ...
    }
    } finally {
    depth--;
    bin.setBlockDataMode(oldMode);
    }
    } private Object readOrdinaryObject(boolean unshared) throws IOException {
    ...
    //isInstantiable 返回true,执行 desc.newInstance(),通过反射创建新的单例类,
    obj = desc.isInstantiable() ? desc.newInstance() : null;
    ...
    // 在Singleton类中添加 readResolve 方法后 desc.hasReadResolveMethod() 方法执行结果为true
    if (obj != null && handles.lookupException(passHandle) == null && desc.hasReadResolveMethod()) {
    // 通过反射调用 Singleton 类中的 readResolve 方法,将返回值赋值给rep变量
    // 这样多次调用ObjectInputStream类中的readObject方法,继而就会调用我们定义的readResolve方法,所以返回的是同一个对象。
    Object rep = desc.invokeReadResolve(obj);
    ...
    }
    return obj;
    }
  • 反射方式破解单例的解决方法

    public class Singleton {
    
        //私有构造方法
    private Singleton() {
    /*
    反射破解单例模式需要添加的代码
    */
    if(instance != null) {
    throw new RuntimeException();
    }
    } private static volatile Singleton instance; //对外提供静态方法获取该对象
    public static Singleton getInstance() { if(instance != null) {
    return instance;
    } synchronized (Singleton.class) {
    if(instance != null) {
    return instance;
    }
    instance = new Singleton();
    return instance;
    }
    }
    }

    说明:

    这种方式比较好理解。当通过反射方式调用构造方法进行创建创建时,直接抛异常。不运行此中操作。

Java中的单利模式的更多相关文章

  1. Java中的单利模式介绍

    单利模式:本来是不准备写的,但是最近发现好多公司面试时都会或多或少的提到单利模式,因此今天把单利模式拉出来说说. 定义:只包含一个被称为单例类的特殊类.通过单例模式可以保证系统中一个类只有一个实例而且 ...

  2. PHP中的抽象类与抽象方法/静态属性和静态方法/PHP中的单利模式(单态模式)/串行化与反串行化(序列化与反序列化)/约束类型/魔术方法小结

      前  言  OOP  学习了好久的PHP,今天来总结一下PHP中的抽象类与抽象方法/静态属性和静态方法/PHP中的单利模式(单态模式)/串行化与反串行化(序列化与反序列化). 1  PHP中的抽象 ...

  3. (转)轻松学,Java 中的代理模式及动态代理

    背景:讲到反射机制,肯定会想到动态代理. 轻松学,Java 中的代理模式及动态代理 代理模式可以在不修改被代理对象的基础上,通过扩展代理类,进行一些功能的附加与增强.值得注意的是,代理类和被代理类应该 ...

  4. Java 设计模式之单利模式

    一.首先介绍一下单例模式:     单例模式(Singleton),也叫单子模式,是一种常用的软件设计模式.在应用这个模式时,单例对象的类必须保证只有一个实例存在.许多时候整个系统只需要拥有一个的全局 ...

  5. Java设计模式之单利模式(Singleton)

    单利模式的应用场景: 单利模式(Singleton Pattern)是指确保一个类在任何情况下都绝对只有一个实例.并提供一个全局反访问点.单利模式是创建型模式.单利模式在生活中应用也很广泛,比如公司C ...

  6. JAVA中的单利

    单列:单例模式分三种:懒汉式单例.饿汉式单例.登记式单例三种.单例模式有一下特点:1.单例类只能有一个实例.2.单例类必须自己自己创建自己的唯一实例.3.单例类必须给所有其他对象提供这一实例. 一.懒 ...

  7. Java中的代理模式

    代理模式在Java Web的框架中经常使用到.比如说在对数据库的访问中,核心功能是对数据库的增删改查,而连接数据库.处理事务等功能我们在开发中也要考虑到.所以我们将数据库的CRUD抽象到接口中,然后实 ...

  8. 说说Java中的代理模式

    今天看到传智播客李勇老师的JDBC系列的第36节——通过代理模式来保持用户关闭连接的习惯.讲的我彻底蒙蔽了,由于第一次接触代理模式,感到理解很难,在博客园找到一篇文章,先记录如下: 引用自java设计 ...

  9. java 设计模式之单利模式以及代理模式(静态)

    1:单利模式: public class Singleton { private static Singleton uniqueInstance = null; private Singleton() ...

随机推荐

  1. python03day

    回顾 pycharm简单使用 while循环 结构 pass while实现打印1-2+3-4+--+99 格式化输出:针对str,让字符串中某些位置变为动态可传入的 % s str d digist ...

  2. 「JOISC 2016 Day 1」棋盘游戏

    「JOISC 2016 Day 1」棋盘游戏 先判无解:第1,3行有连续的空格或四个角有空格. 然后可以发现有解的情况第1,3行可以在任意时间摆放. 对于某一列,若第2行放有棋子,那么显然可以把棋盘分 ...

  3. Java 给Word每一页设置不同图片水印效果

    Word中设置水印时,可加载图片设置为水印效果,但通常添加水印效果时,会对所有页面都设置成统一效果,如果需要对每一页或者某个页面设置不同的水印效果,则可以参考本文中的方法.下面,将以Java代码为例, ...

  4. JspSmartUpload 简略中文API文档

    感谢原文作者:~数字人生~ 原文链接:https://www.cnblogs.com/mycodelife/archive/2009/04/26/1444132.html 一.JspSmartUplo ...

  5. Web标准和骨架

    Web 标准的好处 1.让Web的发展前景更广阔 2.内容能被更广泛的设备访问 3.更容易被搜寻引擎搜索 4.降低网站流量费用 5.使网站更易于维护 6.提高页面浏览速度 Web 标准构成 Web标准 ...

  6. Mybatis返回插入数据的主键的两种方式

    方式一: 需要在映射文件中添加如下片段: <insert id="insertProduct" parameterType="domain.model.Produc ...

  7. Swift 学习网址精选 By HL

    虽然目前iOS大部分的项目开发语言用的不是Swift,但随着Swift的不断强大,取代Objective-C 指日可待,所以学习Swift是十分必要的.但毕竟是亲儿子,目前只有Foundation被翻 ...

  8. 简单的springboot + vue

    安装vue 脚手架 npm install -g @vue/cli 查看vue 版本 vue -V 创建vue项目 vue create vue_project Vue CLI v4.5.13? Pl ...

  9. suse 12 二进制部署 Kubernetets 1.19.7 - 第03章 - 部署flannel插件

    文章目录 1.3.部署flannel网络 1.3.0.下载flannel二进制文件 1.3.1.创建flannel证书和私钥 1.3.2.生成flannel证书和私钥 1.3.3.将pod网段写入et ...

  10. Error from server error dialing backend remote error tls internal error

    # kubectl exec -it mysql-master-8cfb64ff9-ct4dx -n prophet -- /bin/bash Error from server: error dia ...