zoukankan      html  css  js  c++  java
  • Java设计模式(4)——创建型模式之单例模式(Singleton)

    一、概述

      弥补一下之前没有给设计模式下的定义,先介绍一下设计模式(引用自百度百科)

      

    设计模式(Design Pattern)是一套被反复使用、多数人知晓的、经过分类的、代码设计经验的总结。
    使用设计模式的目的:为了代码可重用性、让代码更容易被他人理解、保证代码可靠性。 设计模式使代码编写真正工程化;设计模式是软件工程的基石脉络,如同大厦的结构一样。

      设计模式概念的介绍,参见http://blog.jobbole.com/101076/

      其中涉及的设计原则的概念,参见随笔:http://www.cnblogs.com/jiangbei/p/6910790.html

    二、单例模式

      单例模式:保证整个应用程序中某个类的实例有且只有一个(日志对象等)

      常见的单例分为两种——饿汉式懒汉式

      1.饿汉式

      通过静态属性:

      创建Sinleton类,此类实例只能唯一 

    public class Singleton {
        // 1.构造器私有化,保证不能通过外部创建实例
        private Singleton() {
    
        }
        // 2.自己内部创建唯一实例
        static Singleton instance = new Singleton();
    }

      因为是静态成员,所以可以通过类.的形式访问:

    public class Demo01 {
        public static void main(String[] args) {
            // 静态方法创建两个对象
            Singleton s1 = Singleton.instance;
            Singleton s2 = Singleton.instance;
            if (s1 == s2) {
                System.out.println("s1、s2是同一个实例!");
            } else {
                System.out.println("s1、s2不是同一个实例!");
            } 
        }
    }

      

      通过静态方法:

      考虑到我们Java中非常重要的封装的概念,我们有必要对上述代码进行改进,让属性私有化,使得外部无法直接访问:

      新的Singleton类:

    public class Singleton {
        // 1.构造器私有化,保证不能通过外部创建实例
        private Singleton() {
    
        }
        // 2.创建私有唯一实例
        private static Singleton instance = new Singleton();
        // 3.对外创建公共的静态方法获取实例
        public static Singleton getInstance() {
            return instance;
        }
    }

      测试类:

     public static void main(String[] args) {
            // 静态方法创建两个对象
            Singleton s1 = Singleton.getInstance();
            Singleton s2 = Singleton.getInstance();
            if (s1 == s2) {
                System.out.println("s1、s2是同一个实例!");
            } else {
                System.out.println("s1、s2不是同一个实例!");
            }
        }

      

       以上就是饿汉式的单例模式,正如其名,饿汉饿汉,饿了就要吃饭,所以它早早的(静态,类加载的时候)就创建好了实例

      创建完实例后以后不再改变,所以天生是线程安全的。

      2.懒汉式

      单例类Singleton2:

    public class Singleton2 {
        // 1.构造器私有化,不允许外部直接创建
        private Singleton2() {
    
        }
        // 2.声明静态属性(而不实例化)
        private static Singleton2 instance;
        // 3.对外提供静态获取方法
        public static Singleton2 getInstance() {
            if (instance == null) {
                instance = new Singleton2();
            }
            return instance;
        }
    }

      测试类:

      public static void main(String[] args) {
            // 静态方法创建两个对象
            Singleton2 s1 = Singleton2.getInstance();
            Singleton2 s2 = Singleton2.getInstance();
            if (s1 == s2) {
                System.out.println("s1、s2是同一个实例!");
            } else {
                System.out.println("s1、s2不是同一个实例!");
            }
        }

      

      懒汉式,也是和它的命名一样,懒得创建对象,初始化时只声明,第一次访问时才创建

       当然,不得不指出,上述懒汉式是线程不安全的,多线程下是会存在创建多个实例的问题的,为此我们通过以下的改进方法来保证线程安全:

        1.在getInstance()方法上加上同步(返回值前加):

     public static synchronized Singleton2 getInstance() {
            if (instance == null) {
                instance = new Singleton2();
            }
            return instance;
        }

        2.双重检查double checking

      补充;此种双重检查有待考察,实际中请使用饿汉式!

      public static Singleton2 getInstance() {
            if (instance == null) {
                synchronized (Singleton2.class) {
                    if (instance == null) {
                        instance = new Singleton2();
                    }
                }
            }
            return instance;
        }

      // 如果没有双重检查,少了任何一重,都会出现要么性能不高,要么创建多个实例

       想了解更过有关单例的内容,请参见http://blog.csdn.net/jason0539/article/details/23297037/

      将单例模式中的静态变量(单例)推广到多个静态变量,工厂方法根据参数返回不同的变量,即引申为多例模式!这里不再展开多例模式,可以参阅《Java与模式》一书!

  • 相关阅读:
    2020北航OO第二单元总结
    2020北航OO第一单元总结
    OO结课了,狂喜
    BUAAOO第三单元总结
    BUAAOO第二单元代码分析
    BUAAOO第一单元代码分析
    OO第四次博客作业
    OO第三次博客作业
    OO第二次博客作业
    OO第一次博客作业
  • 原文地址:https://www.cnblogs.com/jiangbei/p/7538342.html
Copyright © 2011-2022 走看看