码迷,mamicode.com
首页 > 其他好文 > 详细

单例模式

时间:2017-03-26 00:00:58      阅读:248      评论:0      收藏:0      [点我收藏+]

标签:多线程   sync   实现   分析   帮助   打印   静态方法   单线程   类对象   

定义: 一个类只能产生一个实例,并且该类提供一个唯一的全局访问点。

说明: 在设计系统时,有的对象,需要保证全局只有一个。例如如下对象(线程池,log对象, 注册表信息对象,性能设置对象,驱动类对象,打印机对象等 )

基本思路:

  1. 私有构造器。保证其他类无法实例化该类。

  2. 静态方法。通过该静态方法能得到该对象的实例。

Java中有以下几种写法:

一. 懒汉(线程不安全)

public class Singleton {

    private static Singleton instance;

    private Singleton() {

    }

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

总结: 这种写法起到了lazy loading的效果,即先判断实例是否存在,不存在的情况,再实例化。但是只能在单线程的情况下使用。在多线程的情况下,假如A线程进入到if(instance == null), 但是B线程已经执行到 instance = new Singleton(), 这种情况下,就会产生产生多个实例,未达到单例的效果。

二. 懒汉(线程安全,同步方法)

public class Singleton {

    private static Singleton instance;

    private Singleton() {

    }

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

总结:通过在方法前加关键字 synchronized来起到线程同步的作用,解决线程不安全。具备lazy loading效果。

缺点:效率太低。每个线程在执行getInstance() 方法时,都会进行同步。实际上,该方法仅仅需要在第一次执行时需要同步,99%情况下不需要同步,所以影响性能。

三. 懒汉(同步代码块)

public class Singleton {
    private static Singleton instance;

    private Singleton() {

    }

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

总结: 同步了代码块,但是仔细分析,多线程下,也会出现 写法一的情况,有可能会产生多个实例。

四. 饿汉(静态变量)

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

    private Singleton() {

    }

    public static Singleton getInstance() {
        return instance;
    }
}

 

优点:写法简单,在类装载阶段就完成实例化(单例模式中,大多数是调用getInstance()方法时类装载)。基于classloader机制,避免线程同步问题

缺点:没有lazy loading效果。如果一直没有使用这个实例,会造成内存的浪费。

五. 饿汉(静态代码块)

public class Singleton {
    private static Singleton instance;

    static {
        instance = new Singleton();
    }

    public static Singleton getInstance() {
        return instance;
    }
}

总结: 和上面的,也就是第三种效果差不多。只是把实例化过程放在了静态代码块中,也是在类装载的时候,执行静态代码块,初始化类的实例。

六. 双重校验锁

public class Singleton {
    private static volatile Singleton instance;

    private Singleton() {

    }

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

总结:因为进行了两次if(instance == null)的检查,保证了线程的安全性。

优点:具有lazy loading效果;线程安全;效率高。

缺点:使用了volatile 关键字,JDK1.5以后才支持。

 七. 静态内部类

public class Singleton {

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

    private Singleton() {

    }

    public static Singleton getInstance() {
        return SingletonKeeper.instance;
    }
}

总结: 利用classloader的机制来保证初始化instance实例时只有一个线程。和写法四 和 写法五,还是有区别的(只要Singleton类被装载,那么instance就会被实例化,也就没有lazy loading效果)。 而此写法,当Singleton类被装载了,instance不一定被实例化,因为SingletonKeeper类没有被主动使用,只有显示通过调用getInstance方法时,才会显示装载SingletonKeeper类,从而实例化instance。

类的静态属性只会在第一次加载类的时候初始化,所以在这里,JVM帮助我们保证了线程的安全性,在类进行初始化时,别的线程是无法进入的。

优点:避免了线程不安全,延迟加载,效率高。

八. 枚举

public enum Singleton {
    INSTANCE;
    public void whateverMethod() {

    }
}

总结: 借助JDK1.5中添加的枚举来实现单例模式。不仅能避免多线程同步问题,而且还能防止反序列化重新创建新的对象。

=====================================================================

 

单例模式

标签:多线程   sync   实现   分析   帮助   打印   静态方法   单线程   类对象   

原文地址:http://www.cnblogs.com/FocusIN/p/6618811.html

(0)
(0)
   
举报
评论 一句话评论(0
登录后才能评论!
© 2014 mamicode.com 版权所有  联系我们:gaon5@hotmail.com
迷上了代码!