单例模式的总结

一.单例模式简介

单例模式最初的定义出现于《设计模式》(艾迪生维斯理, 1994):保证一个类仅有一个实例,并提供一个访问它的全局访问点。如果希望在系统中某个类的对象只能存在一个,单例模式是最好的解决方案。由于这个模式的目的是使一个类只能拥有一个实例,因此需要用一种只允许生成对象类的唯一实例的机制,“阻止”所有想要生成对象的访问。

二.单例模式应用场景

对于某些场景来说,只拥有一个实例是很重要的事情,比如说系统的设置文件,一个打印机同时只能为一台机器进行服务。

三.实现要点

1.单例类只能拥有一个实例,即该实例应为静态的。

2.单例类必须自行创建唯一的实例,即该单例类的构造函数应为私有的。

3.单例类必须向整个体统提供唯一的实例,即该类提供了一个静态的公共方法来获取该单例类的私有对象。

四.三种常见的实现方式及其优缺点

1.懒汉式,优点:(1)资源利用率高;

     缺点:(1)第一次加载时不够快;(2)多线程执行时不必要的同步开销大;

2.双重锁式,优点:(1)资源利用率高;(2)去除了懒汉式中的缺点(2);

      缺点:(2)第一次加载时不够快;

3.饿汉式,优点:(1)线程安全;(2)在类加载的同时新建实例,反应速度快;

     缺点:(1)资源利用率不高;

五.Java代码的实现

1.懒汉式

package com.leonzou69.bean;

public class SingletonPrinterOfLazy {
    //声明SingletonPrinter类的一个实例
    private static SingletonPrinterOfLazy instance = null;

    private String tips;

    //重写构造方法,使SingletonPrinter类不能被new实例化
    private SingletonPrinterOfLazy() {}

    //获取SingletonPrinter类实例的方法,懒汉式
    public static synchronized SingletonPrinterOfLazy getInstance() {
        //首先判断单例类是否被实例化
        if(instance == null) {
            instance = new SingletonPrinterOfLazy();
        }
        return instance;
    }

    public String getTips() {
        return tips;
    }

    public void setTips(String tips) {
        this.tips = tips;
    }

}

2.双重锁式

package com.leonzou69.bean;

public class SingletonPrinterOfDouble {
    //声明SingletonPrinter类的一个实例
    private static SingletonPrinterOfDouble instance = null;

    //重写构造方法,使SingletonPrinter类不能被new实例化
    private SingletonPrinterOfDouble() {}

    private String tips;

    //获取SingletonPrinter类实例方法,双重锁式
    public static SingletonPrinterOfDouble getInstance() {
        //首先判断单例类是否被实例化
        if(instance == null) {
            synchronized (SingletonPrinterOfDouble.class) {
                if(instance == null) {
                    instance = new SingletonPrinterOfDouble();
                }
            }
        }
        return instance;
    }

    public String getTips() {
        return tips;
    }

    public void setTips(String tips) {
        this.tips = tips;
    }
}

3.饿汉式

package com.leonzou69.bean;

public class SingletonPrinterOfHungry {
    //直接初始化一个实例
    private static final SingletonPrinterOfHungry instance = new SingletonPrinterOfHungry();

    private String tips;

    //重写默认构造函数使之不能被new关键字初始化
    private SingletonPrinterOfHungry() {
    }

    //获取SingletonPrinterOfHungry类实例方法,饿汉式
    public static SingletonPrinterOfHungry getInstance() {
        return instance;

    }

    public String getTips() {
        return tips;
    }

    public void setTips(String tips) {
        this.tips = tips;
    }
}

六.测试代码以及运行结果

package com.leonzou69.main;

import com.leonzou69.bean.SingletonPrinterOfDouble;
import com.leonzou69.bean.SingletonPrinterOfHungry;
import com.leonzou69.bean.SingletonPrinterOfLazy;

public class Main {

    public static void main(String[] args) {
        //不能用new关键字进行实例化
//        SingletonPrinterOfLazy sp = new SingletonPrinterOfLazy();
//        sp.tips = "不能用new关键字进行实例化";
//        System.out.println(sp.tips);

        //可以使用类内部的的getInstance()方法获取SingletonPrinterOfLazy懒汉实例
        SingletonPrinterOfLazy spOne = SingletonPrinterOfLazy.getInstance();
        spOne.setTips("我是懒汉式单例打印机");
        System.out.println(spOne.getTips());
        //新建另外一个懒汉实例,获取的依然是之前的懒汉实例
        SingletonPrinterOfLazy spTwo = SingletonPrinterOfLazy.getInstance();
        System.out.println(spTwo.getTips());

        //可以使用类内部的etInstance()方法获取SingletonPrinterOfDouble双重锁式实例
        SingletonPrinterOfDouble spThree = SingletonPrinterOfDouble.getInstance();
        spThree.setTips("我是双重锁式单例打印机");
        System.out.println(spThree.getTips());
        //新建另外一个双重锁实例,获取的依赖是之前的双重锁实例
        SingletonPrinterOfDouble spFour  = SingletonPrinterOfDouble.getInstance();
        System.out.println(spThree.getTips());

        //可以使用类内部的etInstance()方法获取SingletonPrinterOfHungry饿汉实例
        SingletonPrinterOfHungry spFive = SingletonPrinterOfHungry.getInstance();
        spFive.setTips("我是饿汉式单例打印机");
        System.out.println(spFive.getTips());
        //新建另外一个双重锁实例,获取的依赖是之前的双重锁实例
        SingletonPrinterOfHungry spSix  = SingletonPrinterOfHungry.getInstance();
        System.out.println(spSix.getTips());
    }
}

我是懒汉式单例打印机
我是懒汉式单例打印机
我是双重锁式单例打印机
我是双重锁式单例打印机
我是饿汉式单例打印机
我是饿汉式单例打印机

七.单例模式在Android上的一个应用,点击按钮退出所有Activity。

源码地址:https://github.com/leonzou69/DesignPatterns

时间: 2024-09-30 15:06:00

单例模式的总结的相关文章

Java设计模式学习笔记,一:单例模式

开始学习Java的设计模式,因为做了很多年C语言,所以语言基础的学习很快,但是面向过程向面向对象的编程思想的转变还是需要耗费很多的代码量的.所有希望通过设计模式的学习,能更深入的学习. 把学习过程中的笔记,记录下来,只记干货. 第一部分:单例模式的内容 单例模式:类只能有一个实例. 类的特点:1.私有构造器:2.内部构造实例对象:3.对外提供获取唯一实例的public方法. 常见的单例模式实现有五种形式: 1.饿汉式. 2.懒汉式. 3.双重检查锁式. 4.静态内部类式. 5.枚举式. 以下分别

【python之路33】开发模式单例模式

1.单例模式指的是创建单个实例,例如:数据库连接池中包含10个数据库连接,用户访问数据时随机从连接池中拿出一个进行连接,其他用户再次访问时不再创建对象进行连接 #!usr/bin/env python # -*- coding:utf-8 -*- class ConnecttionPool: __instance = None def __init__(self): self.ip = '222.133.177.67' self.port = '3306' self.username = 'od

.Net 单例模式(Singleton)

每台计算机可以有若干个打印机,但只能有一个Printer Spooler, 以避免两个打印作业同时输出到打印机中.每台计算机可以有若干传真卡,但是只应该有一个软件负责管理传真卡,以避免出现两份传真作业同时传到传真卡中的情况.每台计算机可以有若干通信端口,系统应当集中管理这些通信端口,以避免一个通信端口同时被两个请求同时调用. 问题描述: 单例模式 Singleton Pattern 问题解决: (1)单例模式简介: Singleton模式要求一个类有且仅有一个实例,并且提供了一个全局的访问点.这

设计模式(一)----单例模式

单例模式其实就是每次实例化时都会得到一个相同的对象. 单例模式的写法有个简单的口诀:三私一公即一个私有的静态属性,私有的构造方法,私有的克隆方法还有一个公共的静态方法. <?phpclass Cat{ //私有的静态属性 private static $instance; //私有的构造方法 private function __construct(){ echo "这是一个单例模式"; } //公共的静态方法 public static function getInstance

Qt中单例模式的实现(4种方法)

最简单的写法: 12345 static MyClass* MyClass::Instance(){ static MyClass inst; return &inst;} 过去很长一段时间一直都这么写,简单粗暴有效.但是直接声明静态对象会使编译出的可执行文件增大,也有可能出现其他的一些问题,所以利用了Qt自带的智能指针QScopedPointer和线程锁QMutex,改成了需要时才动态初始化的模式: 12345678910111213 static MyClass* MyClass::Inst

设计模式之单例模式

单例模式是软件开发中非常普遍的一种模式.它的主要作用是确保系统中,始终只存在一个类的实例对象. 这样做的好处有两点: 1.对于需要频繁使用的对象,在每次使用时,如果都需要重新创建,并且这些对象的内容都是一样的.则不但提高了jvm的性能开销(堆中开辟新地址,同时降低GC效率等),同时还会降低代码的运行效率.倘若始终在堆中只存在唯一的一个实例对象.任何方法在使用时,均直接访问这个实例对象,则大大提高了系统的运行效率. 2.可以更好的维护对象,倘若系统中存在多个相同的实例对象,而一旦这些实例对象的属性

设计模式之单例模式(c++)

问题描述 Singleton 模式解决问题十分常见, 我们怎样去创建一个唯一的变量( 对象)?在基于对象的设计中我们可以通过创建一个全局变量(对象) 来实现,在面向对象和面向过程结合的设计范式(如C++中)中,我们也还是可以通过一个全局变量实现这一点.但是当我们遇到了纯粹的面向对象范式中,这一点可能就只能是通过Singleton模式来实现了,可能这也正是很多公司在招聘 Java 开发人员时候经常考察Singleton 模式的缘故吧. (全局变量在项目中是能不用就不用的,它是一个定时炸弹,是一个不

设计模式一(单例模式)

单例模式:所谓单例模式就是确保类只有一个对象,并提供一个公共的访问接口.下面根据概念分析单例模式. 1.确保一个实例,就是不让程序随处可以new一个对象,这个怎么实现呢?运用私有构造函数. public class Singleton { private Singleton() { } } 这样Singleton就不能new对象了. 2.唯一的对象在何处?在内部定义一个静态的Singleton对象,为什么要定义静态的呢,因为静态对象属于类.然后提供一个公共的访问接口即可 public class

java/android 设计模式学习笔记(一)---单例模式

前段时间公司一些同事在讨论单例模式(我是最渣的一个,都插不上嘴 T__T ),这个模式使用的频率很高,也可能是很多人最熟悉的设计模式,当然单例模式也算是最简单的设计模式之一吧,简单归简单,但是在实际使用的时候也会有一些坑. PS:对技术感兴趣的同鞋加群544645972一起交流 设计模式总目录 java/android 设计模式学习笔记目录 特点 确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例. 单例模式的使用很广泛,比如:线程池(threadpool).缓存(cache).对

单例模式与多线程

概述 关于一般单例模式的创建和分析在我的另一篇博客<Java设计模式--单件模式>中有详细说明.只是在上篇博客中的单例是针对于单线程的操作,而对于多线程却并不适用,本文就从单例模式与多线程安全的角度出发,讲解单例模式在多线程中应该如何被使用. 版权说明 著作权归作者所有. 商业转载请联系作者获得授权,非商业转载请注明出处. 本文作者:Coding-Naga 发表日期: 2016年4月6日 本文链接:http://blog.csdn.net/lemon_tree12138/article/det