Java设计模式--单例模式

单例模式

饿汉式单例

  1. 自己提供实例
  2. 构造方法私有
  3. 提供静态方法给其他人用
public class Hungry {
    
    private Hungry(){};
    
    private final static Hungry HUNGRY = new Hungry();
    
    public static Hungry getInstance(){
        return HUNGRY;
    }
}


//静态内部类实现
public class Holder {

    private Holder(){};

    public static Holder getInstance(){
        return InnerClass.HOLDER;
    }

    public static class InnerClass{
        private static final Holder HOLDER = new Holder();
    }
}

双重检测锁模式的懒汉式单例 (DCL懒汉式)

public class LazyMan {
    
    private static boolean pei = false;

    private LazyMan() {
        //System.out.println(Thread.currentThread().getName()+"启动");
        //四重检测,预防反射破坏
        synchronized (LazyMan.class){
            if (pei == false){
                pei = true;
            }else {
                throw new RuntimeException("不要试图用反射破坏单例异常");
            }
        }
    }

    //避免指令重排
    private volatile static LazyMan lazyMan;

    //双重检测锁模式的 懒汉式单例 DCL懒汉式
    public static LazyMan getInstance() {
        if (lazyMan==null){
            synchronized (LazyMan.class){
                if (lazyMan == null) {
                    lazyMan = new LazyMan();//不是原子性操作
                    /**
                     * 1、分配内存空间
                     * 2、执行构造方法,初始化对象
                     * 3、把这个对象指向这个空间
                     * 多线程下可能导致指令重排
                     * 比如A线程为132,先把空间占了,但是还没完成构造
                     * 线程B直接return没有构造的对象
                     */
                }
            }
        }
        return lazyMan;
    }

    public static void main(String[] args) {
        for (int i = 0; i < 10; i++) {
            new Thread(()->{
                LazyMan.getInstance();
            }).start();
        }
        
        //反射可以破坏单例,道高一尺魔高一丈
        //LazyMan instance1 = LazyMan.getInstance();
        Field pei = LazyMan.class.getDeclaredField("pei");
        pei.setAccessible(true);
        Constructor<LazyMan> declaredConstructor = LazyMan.class.getDeclaredConstructor(null);
        declaredConstructor.setAccessible(true);
        LazyMan instance1 = declaredConstructor.newInstance();
        pei.set(instance1,false);
        LazyMan instance2 = declaredConstructor.newInstance();
        System.out.println(instance1.hashCode());
        System.out.println(instance2.hashCode());
    }
}

枚举单例(不会被反射破坏)

//枚举本身也是一个Class类
//Cannot reflectively create enum objects
public enum  EnumSingleton {

    INSTANCE;

    public EnumSingleton getInstance(){
        return INSTANCE;
    }

    public static void main(String[] args) throws NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException {
        EnumSingleton instance1 = EnumSingleton.INSTANCE;
        //通过jad反编译,得到EnumSingleton的构造方法,但是还是不能破坏枚举单例
        /*private EnumSingleton(String s, int i)
        {
            super(s, i);
        }*/
        Constructor<EnumSingleton> declaredConstructor = EnumSingleton.class.getDeclaredConstructor(String.class,int.class);
        declaredConstructor.setAccessible(true);
        EnumSingleton instance2 = declaredConstructor.newInstance();
        System.out.println(instance1);
        System.out.println(instance2);
    }
}

在这里插入图片描述

枚举反编译源码

// Decompiled by Jad v1.5.8g. Copyright 2001 Pavel Kouznetsov.
// Jad home page: http://www.kpdus.com/jad.html
// Decompiler options: packimports(3) 
// Source File Name:   EnumSingleton.java

package com.peidan.singleton;

import java.io.PrintStream;
import java.lang.reflect.Constructor;
import java.lang.reflect.InvocationTargetException;

public final class EnumSingleton extends Enum
{

    public static EnumSingleton[] values()
    {
        return (EnumSingleton[])$VALUES.clone();
    }

    public static EnumSingleton valueOf(String name)
    {
        return (EnumSingleton)Enum.valueOf(com/peidan/singleton/EnumSingleton, name);
    }

    private EnumSingleton(String s, int i)
    {
        super(s, i);
    }

    public EnumSingleton getInstance()
    {
        return INSTANCE;
    }

    public static void main(String args[])
        throws NoSuchMethodException, IllegalAccessException, InvocationTargetException, InstantiationException
    {
        EnumSingleton instance1 = INSTANCE;
        Constructor declaredConstructor = com/peidan/singleton/EnumSingleton.getDeclaredConstructor(null);
        declaredConstructor.setAccessible(true);
        EnumSingleton instance2 = (EnumSingleton)declaredConstructor.newInstance(new Object[0]);
        System.out.println(instance1);
        System.out.println(instance2);
    }

    public static final EnumSingleton INSTANCE;
    private static final EnumSingleton $VALUES[];

    static 
    {
        INSTANCE = new EnumSingleton("INSTANCE", 0);
        $VALUES = (new EnumSingleton[] {
            INSTANCE
        });
    }
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值