GOF设计模式——Singleton模式

一、什么是Singleton模式?

        Singleton模式就是平常所说的单例模式,简单来说就只有一个实例。在使用java.lang.String的时候,将不同的字符串赋给String引用,其实就是创建了一个String对象实例,当有1000个不同的字符串创建的时候,就会出现1000个实例。有时候,我们只想在程序里面创建一个实例,譬如Hibernate的SessionFactory,那么我们可以使用Singleton模式来设计某一个类从始至终,程序调用的都是属于同一个实例。

二、Singleton模式的思想

实现单例的类里面有一个私有的本类对象属性;一个私有的构造方法,说明外面的类不可以通过new的方式来创建该对象;还有一个public的getInstance方法,用于给外面的类调用获取Singleton对象实例。

三、具体实例

        实现Singleton模式有多种方式,主要分为两大类:懒汉模式饿汉模式

        所谓的懒汉模式,就是比较“懒”,需要调用的时候,才去加载资源(创建实例);饿汉模式就跟懒汉模式相反,在程序启动,类加载的时候就顺便加载实现了。

1、懒汉模式(最常见,也最安全高效)

package com.cjs.Singleton;
 
public class SingletonLazy {
    private static SingletonLazy singletonLazy;
 
    private SingletonLazy() {
    }
 
    public static SingletonLazy getInstance() {
        if (singletonLazy == null) {
            synchronized (SingletonLazy.class) {
                if (singletonLazy == null) {
                    singletonLazy = new SingletonLazy();
                }
            }
        }
        return singletonLazy;
    }
}

为什么说是最常见,也最安全高效?因为懒汉模式可以合理利用分配资源,而且使用synchronnized锁可以保证创建实例的时候不被其他线程占用。

编写一个Main类,运行一下:

package com.cjs.Singleton;
 
public class Main {
    public static void main(String[] args) {
 
        Runnable r = new Runnable() {
            @Override
            public void run() {
               SingletonLazy singletonLazy = SingletonLazy.getInstance();
                System.out.println(singletonLazy.hashCode());
            }
        };
        for (int i = 0; i < 5; i++) {
            try {
                Thread t = new Thread(r);
                t.start();
                Thread.sleep(2000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
 
    }
}

输出:


它们的hascode都是一样的值,说明它们获取的实例都是同一个。

2、饿汉模式

饿汉模式实现Singleton模式,有分类好几种:静态生成;静态内之类生成;序列化与反序列化;枚举实现;

他们实现的原理都一样,都是程序启动的时候,就进行加载创建实例。下面举个简单的例子:

package com.cjs.Singleton;
 
public class SingletonStatic {
    private static class MySingletonStatic{
        private static SingletonStatic singletonStatic = new SingletonStatic();
    }
 
    private SingletonStatic() {
 
    }
 
    public static SingletonStatic getInstance() {
        return MySingletonStatic.singletonStatic;
    }
}

Main类:

package com.cjs.Singleton;
 
public class Main {
    public static void main(String[] args) {
 
        Runnable r = new Runnable() {
            @Override
            public void run() {
//               SingletonLazy singletonLazy = SingletonLazy.getInstance();
                SingletonStatic singletonStatic = SingletonStatic.getInstance();
                System.out.println(singletonStatic.hashCode());
            }
        };
        for (int i = 0; i < 5; i++) {
            try {
                Thread t = new Thread(r);
                t.start();
                Thread.sleep(2000);
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
 
    }
}

输出:


posted @ 2019-01-27 16:17  KamShing  阅读(287)  评论(0编辑  收藏  举报