设计模式(4)之单例模式

1. 什么是单例模式
单例模式是一种对象创建型模式,使用单例模式,可以保证为一个类只生成唯一的实例对象。也就是说,在整个程序空间中,该类只存在一个实例对象。 其实,GoF对单例模式的定义是:保证一个类、只有一个实例存在,同时提供能对该实例加以访问的全局访问方法。

2. 为什么要使用单例模式呢?
在应用系统开发中,我们常常有以下需求:
- 在多个线程之间,比如servlet环境,共享同一个资源或者操作同一个对象
- 在整个程序空间使用全局变量,共享资源
- 大规模系统中,为了性能的考虑,需要节省对象的创建时间等等。

因为Singleton模式可以保证为一个类只生成唯一的实例对象,所以这些情况,Singleton模式就派上用场了。

3. 单例模式实现
1.饿汉式。
2.懒汉式。
3.双重检查。

//Perosn.java

//一般的类
public class Person {

    private String name;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

}

使用的时候,直接new。

//MainClass.java

public class MainClass {

    public static void main(String[] args) {

        Person person1 = new Person();
        Person person2 = new Person();

        person1.setName("zhangsan");
        person2.setName("lisi");

        System.out.println(person1.getName());
        System.out.println(person2.getName());
    }
}

显然,person1和person2是两个不同的对象,打印的结果肯定不同。所以,要想实单例模式,需要对Person类进行改造。

下面介绍【饿汉式】实现单例模式。

//Person2.java 

//饿汉式实现

public class Person2 {
    // 饿汉式的特点,不管三七二十一,类加载的时候就new一个对象出来,这个对象是全局的,整个应用程序中只有一份。在类加载的时候就new出来。
    private static final Person2 person = new Person2();
    //首先,将构造方法写成私有的,这样在外部就不能new了。
    private Person2(){}

    private String name;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }
    //对外提供一个方法访问这个全局唯一的一个实例对象。
    public static Person2 getPerson2(){
        return person;
    }
}

下面是【懒汉式】实现单例模式:


public class Person3 {

    private static Person3 person = null;

    private Person3(){}

    private String name;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public static **synchronized** Person3 getPerson3(){
        //在获取该类实例之前,如果之前没有new这个实例,则new一个。如果有了这个实例,直接返回之前的那个实例对象。
        //这种在用到的时候才new对象方式就叫做【懒汉式】。
        if(person == null)
            person  = new Person3();

        return person;
    }

}

好了,到这里,是时候说一个问题了。大家有没有注意到,在懒汉式实现方法中,加了一个synchronized关键字,这是线程同步用的。
这里写图片描述
(1)如果没有synchronized关键字
在多线程程序中,如果线程1和2同时进入到getPerson3方法中,又如果线程1执行完if(person==null)之后,此时CPU的执行权切回到线程2,线程2也进入getPerson3方法,执行了if(person==null),因为此时并没有Person3的实例,所以线程1和2都会进如到if语句块中,这样就会导致new了两次Person3,这样肯定不是单利模式啦。
(2)如果加了synchronized关键字
这里写图片描述
如果加了synchronized关键字,那么当线程1访问getPerson3方法的时候,即使CPU执行权被切换到其他线程上,其他线程也不能够访问getPerson3方法,因为该方法当前正在被线程1访问,除非线程1执行完了该方法并退出之后,其他线程才能访问该方法。然而此时已经创建了一个Person3的实例了,person变量不为null,所以尽管其他方法进入getPerson3方法,判断条件不成立,也就不会再创建Person3对象了。这就保证了单例。

但是,大家有没有考虑一个问题,你线程1在访问的时候,其他的线程如果访问该方法都得在外面等着,凭啥呀?这不耽误大家的时间吗,是不是。显然这种方法是不高效的。

我们仔细观察下getPerson3这个方法,这个方法是不是只有一行代码需要同步,就是new Person3这行代码,其他的代码是不需要同步的。所以说呀,为了这行代码,就让整个方法同步,这个不太好吧,如果方法中还有其他一大坨的代码不需要同步呢,你把其他线程拒之门外是不是太耽误事儿了。所以,我们可以使用同步局部代码块的方法解决这个问题。
这里写图片描述
现在我们来分析一下,如果两个线程同时进入到getPerson3方法中,并且都进入到if语句块中,会不会new出两个Person对象呢?不会的。仔细看看代码,当线程1 new了Person之后,person变量就不为null,此时线程2进入同步代码块后,再new之前又进行了一次person是否为null的判断,显然person是不为null,所以肯定不会new一个新的Person,这也保证了单例。这就是第三种实现单例的方式–【双重检查】。

说完【懒汉式】实现方法在多线程中可能出现的问题及解决方法,那么,【饿汉式】会在多线程程序中出现类似上面所说的创建多个实例的问题吗?亲,不会的,因为【饿汉式】随着类加载完毕对象就已经new出来了,不管你多少个线程访问该类中的方法,永远只返回这个已经被创建出来的对象。

小结:
1. 一般懒汉式比饿汉式号效率高,因为【饿汉式】用的时候才new,明智呀。
2. 【懒汉式】实现过程中要注意多线程访问的问题,使用【双重检查】既可以保证多线程之间的同步,又相对来说提高了效率。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值