【算法题】单例模式的8种实现方式(java版)「建议收藏」

【算法题】单例模式的8种实现方式(java版)「建议收藏」根据马士兵老师的视频整理下来的8种单例模式的实现方式,在此记录一下。代码示例1:饿汉式packagecom.examples.singleton;publicclassMgr01{publicstaticvoidmain(String[]args){Mgr01m1=Mgr01.getInstance();Mgr01m2=Mgr01.getInstance();System.out.println(m1…

大家好,又见面了,我是你们的朋友全栈君。

根据马士兵老师的视频整理下来的8种单例模式的实现方式,在此记录一下。

代码示例1:饿汉式

package com.examples.singleton;

public class Mgr01 {
    public static void main(String[] args) {
        Mgr01 m1 = Mgr01.getInstance();
        Mgr01 m2 = Mgr01.getInstance();
        System.out.println(m1==m2);
    }

    //class文件加载时进行类的实例化
    private static Mgr01 INSTANCE = new Mgr01();
    //构造方法私有化
    private Mgr01(){};
    public static Mgr01 getInstance(){return INSTANCE;}
    public void m(){ System.out.println("Mgr01"); }
}

说明:

以上是饿汉式单例,就是在类加载的时候完成实例化,而且只有一个实例;同时JVM能够保证线程安全,简单实用,推荐使用。

缺点:不管是否用到,类装载的时候就已经完成了实例化

代码示例2:和示例1类似,只不过将实例化在静态代码块中完成

package com.examples.singleton;


public class Mgr02 {
    public static void main(String[] args) {
        Mgr02 m1 = Mgr02.getInstance();
        Mgr02 m2 = Mgr02.getInstance();
        System.out.println(m1==m2);
    }

    //class文件加载时进行类的实例化
    private static Mgr02 INSTANCE ;
    static {

        INSTANCE = new Mgr02();
    }

    //构造方法私有化
    private Mgr02(){};
    public static Mgr02 getInstance(){return INSTANCE;}
    public void m(){ System.out.println("Mgr01"); }

}

代码示例3:懒汉式

package com.examples.singleton;


public class Mgr03 {
    public static void main(String[] args) {
        for(int i=0;i<50;i++) {
            new Thread(()->{
               System.out.println(Mgr03.getInstance().hashCode());
            }).start();
        }
    }

    private static Mgr03 INSTANCE ;
    //构造方法私有化
    private Mgr03(){};
    public static Mgr03 getInstance(){
        if(INSTANCE==null){
            try{
                Thread.sleep(1);
            }catch (InterruptedException e){
                e.printStackTrace();
            }
            INSTANCE = new Mgr03();
        }
        return INSTANCE;
    }
    public void m(){ System.out.println("Mgr01"); }
}

多线程执行结果:

【算法题】单例模式的8种实现方式(java版)「建议收藏」

说明:为了解决不管是否使用都会实例化的问题,但是却带来线程不安全的问题

代码示例4:懒汉式+锁

package com.examples.singleton;


public class Mgr03 {
    public static void main(String[] args) {
        for(int i=0;i<50;i++) {
            new Thread(()->{
                System.out.println(Mgr03.getInstance().hashCode());
           }).start();
        }
    }

    private static Mgr03 INSTANCE ;
    //构造方法私有化
    private Mgr03(){};
    public static synchronized Mgr03 getInstance(){
        if(INSTANCE==null){
            try{
                Thread.sleep(1);
            }catch (InterruptedException e){
                e.printStackTrace();
            }
            INSTANCE = new Mgr03();
        }
       return INSTANCE;
    }
    public void m(){ System.out.println("Mgr01"); }
}

说明: 通过给getInstance()方法加锁,解决对象不唯一的问题,但是同时也带来效率下降的问题

代码示例5:懒汉式+锁+减少同步代码块

package com.examples.singleton;


public class Mgr05 {
    public static void main(String[] args) {
        for(int i=0;i<50;i++) {
            new Thread(()->{
                System.out.println(Mgr05.getInstance().hashCode());
            }).start();
        }
    }

    private static Mgr05 INSTANCE ;
    private Mgr05(){};
    public static  Mgr05 getInstance(){
        if(INSTANCE==null){
            //试图通过减小同步代码块的方式提高效率,但是并不可行
            synchronized(Mgr05.class){
                try{
                    Thread.sleep(1);
                }catch (InterruptedException e){
                   e.printStackTrace();
                }
                INSTANCE = new Mgr05();
            }
        }
        return INSTANCE;
    }
    public void m(){ System.out.println("Mgr01"); }

}

多线程执行结果:

【算法题】单例模式的8种实现方式(java版)「建议收藏」

说明:同样存在线程不安全的问题,因为第一个if判断条件有可能导致在对象没有实例化之前,不同的线程进入到if代码块中,拿到锁之后进行实例化

代码示例6:懒汉式+锁+双重检查

package com.examples.singleton;


public class Mgr06 {
    public static void main(String[] args) {
        for(int i=0;i<50;i++) {
            new Thread(()->{
                System.out.println(Mgr06.getInstance().hashCode());
            }).start();
        }
    }

    private static Mgr06 INSTANCE ;
    private Mgr06(){};
    public static Mgr06 getInstance(){
        if(INSTANCE==null){
            //双重检查
            synchronized(Mgr06.class){
                if(INSTANCE==null) {
                    try{
                        Thread.sleep(1);
                    }catch (InterruptedException e){
                        e.printStackTrace();
                    }
                    INSTANCE = new Mgr06();
                }
            }
        }
        return INSTANCE;
    }
    public void m(){ System.out.println("Mgr01"); }
}

可以保证线程安全

代码示例7:静态内部类

package com.examples.singleton;


public class Mgr07 {
    public static void main(String[] args) {
        for(int i=0;i<50;i++) {
            new Thread(()->{
                System.out.println(Mgr07.getInstance().hashCode());
            }).start();
        }
    }
    private static  class Mgr07Holder{
        private final static Mgr07 INSTANCE = new Mgr07();
    }
    private Mgr07(){};
    public static Mgr07 getInstance(){
        return Mgr07Holder.INSTANCE;
    }
    public void m(){ System.out.println("Mgr01"); }
}

说明:加载外部类时不会加载内部类,这样可以实现懒加载,JVM保证线程安全,类只加载一次

代码示例8:枚举类

package com.examples.singleton;


public enum Mgr08 {
    INSTANCE;
    public void m(){ System.out.println("Mgr08"); }
    public static void main(String[] args) {
        for(int i=0;i<50;i++) {
            new Thread(()->{
                System.out.println(Mgr08.INSTANCE.hashCode());
            }).start();
        }
    }
}

说明:以上方法不单能实现单例,而且能防止被反序列化,因为枚举类没有构造方法,所以即使得到了class文件也不能被反序列化。

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/156766.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • 八种排序算法的时间复杂度复杂度

    八种排序算法的时间复杂度复杂度https://www.cnblogs.com/dll-ft/p/5861210.html 转载1、稳定性归并排序、冒泡排序、插入排序。基数排序是稳定的选择排序、快速排序、希尔排序、堆排序是不稳定的 2、时间复杂度最基础的四个算法:冒泡、选择、插入、快排中,快排的时间复杂度最小O(n*log2n),其他都是O(n2)排序法 平均时间 最差情形 稳定度…

    2022年5月14日
    76
  • 分苹果Python_c++递归

    分苹果Python_c++递归题目《分苹果》的Python解决方案。

    2022年10月12日
    1
  • SourceInsight3注册码[通俗易懂]

    SourceInsight3注册码[通俗易懂]SI3US-241109-94280

    2022年10月3日
    0
  • 创建线程池的七种方式为_全局线程池如何创建

    创建线程池的七种方式为_全局线程池如何创建创建线程池的7种方式

    2022年10月1日
    0
  • Java 递归、DFS、回溯

    Java 递归、DFS、回溯DFS/回溯算法如果某问题的解可以由多个步骤得到,而每个步骤都有若干种选择(这些候选方案集可能会依赖之前做出的选择),且可以用递归枚举法实现,则它的工作方式可以用解答树来描述。全排列问题输出数字1~N所能组成的所有全排列publicclassA{/***全排列**@paramargs*/staticVector<Integer>vector=newVector<>();s

    2022年7月8日
    15
  • node读取文件操作

    node读取文件操作导言:nodejs中所有与文件相关的操作都在fs模块中,而读写操作又是我们会经常用到的操作,nodejs的fs模块针对读操作为我们提供了readFile,read,createReadStream三个方法,针对写操作为我们提供了writeFile,write,createWriteStream三个方法,下面分析一下它们的区别:一、readFile和writeFile1、readFile…

    2022年5月29日
    46

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号