单例模式的各种实现

单例模式的主要优点如下: (1)    单例模式提供了对唯一实例的受控访问。因为单例类封装了它的唯一实例,所以它可以严 格控制客户怎样以及何时访问它。 (2)    由于在系统内存中只存在一个对象,因此可以节约系统资源,对于一些需要频繁创建和销 毁的对象单例模式无疑可以提高系统的性能。 (3)    允许可变数目的实例。基于单例模式我们可以进行扩展,使用与单例控制相似的方法来获 得指定个数的对象实例,既节省系统资源,又解决了单例单例对象共享过多有损性能的问 题。 2.主要缺点 单例模式的主要缺点如下: (1)    由于单例模式中没有抽象层,因此单例类的扩展有很大的困难。 (2)    单例类的职责过重,在一定程度上违背了“单一职责原则”。因为单例类既充当了工厂角 色,提供了工厂方法,同时又充当了产品角色,包含一些业务方法,将产品的创建和产品的 本身的功能融合到一起。 (3)    现在很多面向对象语言(如Java、C#)的运行环境都提供了自动垃圾回收的技术,因此,如 果实例化的共享对象长时间不被利用,系统会认为它是垃圾,会自动销毁并回收资源,下次 利用时又将重新实例化,这将导致共享的单例对象状态的丢失。 3.适用场景 在以下情况下可以考虑使用单例模式: (1)    系统只需要一个实例对象,如系统要求提供一个唯一的序列号生成器或资源管理器,或者 需要考虑资源消耗太大而只允许创建一个对象。 (2)    客户调用类的单个实例只允许使用一个公共访问点,除了该公共访问点,不能通过其他途 径访问该实例。 思考 如何对单例模式进行改造,使得系统中某个类的对象可以存在有限多个,例如两例或三例? 【注:改造之后的类可称之为多例类。】

饿汉模式:

public class Esingleton {

    private Esingleton() {

    }

    private static Esingleton ins = new Esingleton();

    public static Esingleton Instance() {
        return ins;
    }


    public static void main(String[] args) {
        Esingleton esingleton = Esingleton.Instance();
        Esingleton esingleton1 = Esingleton.Instance();
        System.out.println(esingleton == esingleton1);
    }
}

懒汉模式:

public class Lsingleton {

    private Lsingleton() {

    }

    private static Lsingleton instance = null;

    public static Lsingleton Instance() {
        if (instance == null)
            instance = new Lsingleton();
        return instance;
    }


    public static void main(String[] args) {
        Lsingleton lsingleton = Lsingleton.Instance();
        Lsingleton lsingleton1 = Lsingleton.Instance();
        System.out.println(lsingleton == lsingleton1);
    }
}

IoDH(    Initialization    Demand    Holder ) 模式:

public class Singleton {

    private Singleton() {
    }

    private static class InnerClass {
        private final static Singleton instance = new Singleton();
    }

    public static Singleton Instance() {
        return InnerClass.instance;
    }

    public static void main(String[] args) {
        Singleton singleton = Singleton.Instance();
        Singleton singleton1 = Singleton.Instance();

        System.out.println(singleton == singleton1);
    }
}

多线程下改造懒汉模式:

public class Lsingleton {

    private Lsingleton() {

    }

    /**
     * 变量对所有线程共享
     */
    private volatile static Lsingleton instance = null;

    public static Lsingleton Instance() {
        //双重锁校验
        if (instance == null)
            synchronized (Lsingleton.class) {
                if (instance == null)
                    instance = new Lsingleton();
            }
        return instance;
    }


    public static void main(String[] args) {
        Lsingleton lsingleton = Lsingleton.Instance();
        Lsingleton lsingleton1 = Lsingleton.Instance();
        System.out.println(lsingleton == lsingleton1);

    Queue<Test> queue = new ArrayBlockingQueue<Test>(100);
        List<Thread> list = new ArrayList<>();
        for (int i = 0; i < 100; i++) {
            list.add(new Thread(() -> {                 
                queue.add(Test.getInstance());
            }));
        }
        for (Thread thread : list) {
            thread.start();
        }

        while (queue.size() != 100);
            System.out.println(""+JSON.toJSONString(queue));
    }
}

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏向治洪

单例模式

单例对象(Singleton)是一种常用的设计模式。在Java应用中,单例对象能保证在一个JVM中,该对象只有一个实例存在。 这样的模式有几个好处: 1、某些...

1717
来自专栏小筱月

ES6 中 Promise 详解

Promise,简单说就是一个容器,里面保存着某个未来才会结束的事件(通常是一个异步操作)的结果。从语法上说,Promise 是一个对象,从它可以获取异步操作的...

832
来自专栏大内老A

WCF技术剖析之二十一:WCF基本异常处理模式[中篇]

通过WCF基本的异常处理模式[上篇], 我们知道了:在默认的情况下,服务端在执行某个服务操作时抛出的异常(在这里指非FaultException异常),其相关的...

19010
来自专栏大内老A

WCF技术剖析之二十七: 如何将一个服务发布成WSDL[编程篇]

对于WCF服务端元数据架构体系来说,通过MetadataExporter将服务的终结点导出成MetadataSet(参考《如何导出WCF服务的元数据》),仅仅是...

1767
来自专栏奔跑的蛙牛技术博客

分布式对象之客户\服务器角色以及RMI初探

程序员希望通过无所不在的java对象来作为所有问题的解决之道,思想就是通过网络请求相互协作。我可以调用一个远程对象帮我得到我所要的信息,并作为响应的一部分返回

551
来自专栏WeTest质量开放平台团队的专栏

Unity3d底层数据传递分析

这篇文章主要分析了在Mono框架下,非托管堆、运行时、托管堆如何关联,以及通过哪些方式调用。内存方面,介绍了什么是封送,以及类和结构体的关系和区别。

752
来自专栏向治洪

Java的依赖注入(控制反转)

两个主角“依赖注入”和“控制反转”: 1、二都说的都是同一件事,只是叫法不同。是一个重要的面向对象编程的法则,也是一种设计模式; 2、英文原称:依赖注入,...

19410
来自专栏WeTest质量开放平台团队的专栏

Unity3d底层数据传递分析

这篇文章主要分析了在Mono框架下,非托管堆、运行时、托管堆如何关联,以及通过哪些方式调用。内存方面,介绍了什么是封送,以及类和结构体的关系和区别。

5242
来自专栏非典型程序猿

从源码透析gRPC调用原理

gRPC是如何work的,清楚的理解其调用逻辑,对于我们更好、更深入的使用gRPC很有必要。因此我们必须深度解析下gRPC的实现逻辑,在本文中,将分别从客户端和...

4637
来自专栏智能算法

你不可不知的23种设计模式

目录 创建型 1. Factory Method(工厂方法) 2. Abstract Factory(抽象工厂) 3. Builder(建造者) 4. Prot...

3276

扫码关注云+社区