设计模式之单例模式

1.饿汉式

package cn.qqa.single_instance;
import java.awt.datatransfer.StringSelection;
/**
 *  饿汉式
 * 很饿,一开始就new出来
 */
public class Hungry {

    //可能会浪费内存空间
    byte [] data1 = new  byte[1024*1024];
    byte [] data2 = new  byte[1024*1024];
    byte [] data3 = new  byte[1024*1024];
    byte [] data4 = new  byte[1024*1024];


    private Hungry(){

    }

    private final static Hungry HUNGRY = new Hungry();

    public static Hungry getInstance(){
        return HUNGRY;
    }

}

2.懒汉式+双重检测锁懒汉式

package cn.qqa.single_instance;

import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.InvocationTargetException;

/**
 *
 * 懒汉式
 *  道高一尺,魔高一丈
 */
public class LazyMan {
    //红绿灯
    private static boolean qqa = false;


    private LazyMan(){
        synchronized (LazyMan.class){
            //用反射获得多个对象
            if(qqa==false){
                qqa=true;
            }else{
                throw new RuntimeException("不要试图通过反射破坏单例模式的异常!");
            }
/*          一个用getInstance,另一个用反射
            if(lazyMan!=null){
                throw new RuntimeException("不要试图通过反射破坏单例模式的异常!");
            }*/
        }

        System.out.println(Thread.currentThread().getName() + "ok");
    }

    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.把这个对象指向这个空间
                     *
                     * 指令重排可能导致的现象:
                     * 正确顺序:123
                     * 132 线程A执行到3,然后线程B会判断lazyMan!=null直接返回,这时对象还没有初始化
                     */
                }
            }
        }

        return lazyMan;
        //单线程下是单例的
    }

    //多线程并发
/*    public static void main(String[] args) {
        for(int i = 0 ; i < 10 ;i++){
            new Thread(()->{
                LazyMan.getInstance();
            }).start();
        }
    }*/

    //反射!可以破坏单例模式!
    public static void main(String[] args) throws Exception {
        //LazyMan instance = LazyMan.getInstance();
        //通过反射获得私有构造器
        Constructor<LazyMan> declaredConstructor = LazyMan.class.getDeclaredConstructor(null);
        //无视私有构造器
        declaredConstructor.setAccessible(true);
        //通过反射获得私有变量
        Field qqa = LazyMan.class.getDeclaredField("qqa");
        qqa.setAccessible(true);
        LazyMan lazyMan = declaredConstructor.newInstance();

        qqa.set(lazyMan,false);

        LazyMan lazyMan1 = declaredConstructor.newInstance();
        System.out.println(lazyMan1.hashCode());
        System.out.println(lazyMan.hashCode());
        System.out.println(lazyMan1==lazyMan);
    }
}

3.静态内部类

package cn.qqa.single_instance;

/**
 *
 * 静态内部类
 * 单例模式必须构造器私有
 */
public class Holder {
    private Holder(){

    }

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

4.枚举类型

package cn.qqa.single_instance;

import java.lang.reflect.Constructor;

//enum是一个什么?
//enum本身也是一个Class类
public enum EnumSingle {
    INSTANCE;
    public EnumSingle getInstance(){
        return INSTANCE;
    }
}

class Test{

    public static void main(String[] args) throws Exception {
        EnumSingle instance = EnumSingle.INSTANCE;

        Constructor<EnumSingle> declaredConstructor = EnumSingle.class.getDeclaredConstructor(null);
        declaredConstructor.setAccessible(true);
        EnumSingle enumSingle = declaredConstructor.newInstance();

        //java.lang.NoSuchMethodException: cn.qqa.single_instance.EnumSingle.<init>()
        System.out.println(instance.hashCode());
        System.out.println(enumSingle.hashCode());


    }
}

 

 

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值