1.设计模式-单例模式

本文介绍了设计模式中的单例模式,详细阐述了实现单例模式的三种方式:懒汉式、饿汉式和枚举式。懒汉式通过双重检验锁保证线程安全;饿汉式在类加载时即初始化实例;枚举式则是一种简洁且线程安全的实现方式,同时避免了序列化问题。
摘要由CSDN通过智能技术生成

实现单例模式的三种方式

单例模式一般分为【懒汉式、饿汉式、枚举式】

懒汉式

package com.example.demo.singleton;
/**
 * 懒汉式-单例
 */
public class LzSingleton {
    // 线程非安全的
//    private static LzSingleton lzSingleton;

    public static String a = "a";
	
    private static volatile LzSingleton lzSingleton;
    private LzSingleton(){}

    // 线程非安全的
//    public static LzSingleton getInstance(){
//        if (lzSingleton == null){
//            lzSingleton = new LzSingleton();
//        }
//        return lzSingleton;
//    }
    private static final class HolderSingleton{
        private static LzSingleton singleton = new LzSingleton();
    }

    // 线程安全 -- 双重检验锁
    public static LzSingleton getInstance(){
        if (lzSingleton == null){
            synchronized(LzSingleton.class){
                if (lzSingleton == null){
                    lzSingleton = new LzSingleton();
                }
            }
        }
        return lzSingleton;
    }

    public static LzSingleton getHolderInstance(){
        return HolderSingleton.singleton;
    }
    
    // 防止序列化、反序列化实例对象
    public static Object readResolve(){
        return lzSingleton;
    }

    public void print(){
        System.out.println("懒汉式");
    }
}

其中注释部分为非线程安全的写法。

  1. getInstance()是其中一种线程安全的写法,采用的是双重检验锁,既保证了线程安全,又保证了执行效率。
  2. getHolderInstance() 采用的是内部类形式,相当于即调用内部类就变成一个饿汉式的形式。
  3. readResolve() 是防止序列化后new LzSingleton(),如果类包含readResolve的方法名,则序列化之后获取的实例就是从这里面返回的。

饿汉式

package com.example.demo.singleton;
/**
 * 饿汉式 -- 单例
 */
public class Singleton {
    public static String a = "a";
//    private static Singleton singleton = new Singleton();
    private static Singleton singleton = null;
    private Singleton(){}
    static {
        if (singleton == null){
            singleton = new Singleton();
        }
    }
    public static Singleton getInstance(){
        return singleton;
    }

    public static Object readResolve(){
        return singleton;
    }

    public static void print(){
        System.out.println("饿汉式");
    }
}

  1. 饿汉式这个写法是把实例对象放在静态代码块中,只要调用这个类里的任何变量,方法,即会强行实例对象。
  2. readResolve()和懒汉式的效果一样

枚举式

我们可以先看一下源代码

package com.example.demo.singleton;
/**
 * 枚举单例
 */
public enum EnumSingleton {
    INSTANCE;
    public void print(){
        System.out.println("枚举式单例");
    }
}

然后再看一下编译后的代码 javap -p EnumSingleton

public final class EnumSingleton extends java.lang.Enum<EnumSingleton> {
  public static final EnumSingleton INSTANCE;
  private static final EnumSingleton[] $VALUES;
  public static EnumSingleton[] values();
  public static EnumSingleton valueOf(java.lang.String);
  private EnumSingleton(); 
  public void print();
  static {};
}

可以看到一个枚举经过编译后,它就转换成一个最终类并且继承了java.lang.Enum
而且这个编译后的类,他有一个private(私有)的构造方法,我们在枚举里写得INSTANCE编译后变成一个public static final EnumSingleton INSTANCE(公开静态常量),因为是final修饰的,所以它只能被实例化一次,并且是在static{}静态代码块中进行实例的(饿汉式),而且枚举不能被序列化。
重点是,重点是,重点是,枚举写单例代码极为简洁。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值