java 设计模式之单例模式

-------Success is getting what you want, happiness is wanting what you get.

 

java设计模式之单例模式(Singleton)

     ---应用场合:只需要一个对象就可以了,例如一个朝代一个皇帝。

     ---作用:保证某个应用程序中某个实例有且只有一个。共同使用一个,节省资源。

     ---类型:饿汉模式,懒汉模式。

一.饿汉模式(类加载的时候就会实例化,线程安全)

 1 package com.sun.singleton;
 2 
 3 //饿汉模式
 4 public class Singleton {
 5 
 6     //将构造函数私有化,不允许外部直接调用创建对象
 7     private Singleton(){
 8         
 9     }
10     //内部创建类的唯一实例,用private static 修饰(加载类时实例化一次,之后无论调用几次用的都是这一个实例)
11     private static Singleton instance = new Singleton();
12     
13     //提供一个外部用于获取实例的方法,用 public static 修饰
14     public static Singleton getInstance(){
15         return instance;
16     }
17 
18 }

二. 懒汉模式(比较懒,所以只有调用方法getInstance时候根据判断结果才会实例化,而且线程不安全)

        线程不安全原因:如果是单线程那懒汉模式也没有问题,但如果是多线程且没有加锁处理就不行了。不如A,B两个线程,A线程首先调用getInstance方法,判断instance 为空,即将创建实例,就在这时cpu切换到B线程,A暂停。B线程调用getInstance方法判断instance为空,然后创建实例,创建结束后B线程结束,A线程开始继续执行,注意:此时A线程已经判断过instance为空,那么会继续创建实例,这样线程AB都各自创建了实例,就不再是单例模式。------所以懒汉模式要加锁同步。

package com.sun.singleton;
//懒汉模式(线程不安全)
public class Singleton2 {
    
    //1.构造函数私有化,不允许外部直接调用
    private Singleton2(){
        
    }
    
    //2.只是声明类的唯一实例,这时没有真正实例化,使用private static 修饰
    private static Singleton2 instance;
    
    //3.提供获取实例的方法,用public static修饰

    // (1).解决线程不安全-------加同步(每个线程每次获取实例都要判断下锁,效率比较低)
    public static synchronized Singleton2 getInstance(){
        if(instance == null){
            return instance = new Singleton2();
        }
        return instance;
    }
    
    //(2).解决线程不安全-----双重检查锁定(提高效率)
    public static Singleton2 getInstance2(){
        /*如果第一个线程获取到了单例的实例对象,后面的线程再获取实例的时候不需要进入同步代码块中了*/
        if(instance == null){
            synchronized (Singleton2.class) {
                if(instance == null){
                    instance = new Singleton2();
                }
            }
        }
        return instance;
    }
    
    
}

 输出:

package com.sun.singleton;

public class Main {

    public static void main(String[] args) {
        
        //实例化两次作比较测试
        //饿汉式
        Singleton s1= Singleton.getInstance();
        Singleton s2= Singleton.getInstance();
        if(s1==s2){
            System.out.println("饿汉式是同一个实例");
        }else{
            System.out.println("饿汉式不是同一个实例");
        }
        //懒汉式
        Singleton2 s3 = Singleton2.getInstance();
        Singleton2 s4 = Singleton2.getInstance();
        if(s3==s4){
            System.out.println("懒汉式是同一个实例");
        }else{
            System.out.println("懒汉式不是同一个实例");
        }

    }

}

 结果截图:

说明单例模式是同一个对象实例化是共享一个内存空间。

两者区别:

           饿汉模式:类加载的时候比较慢(加载时候实例化),但是运行时候比较快,线程安全。

           懒汉模式:类加载的时候比较快,但是运行时候比较慢(调用方法的时候实例化),线程不安全。

这是自己学习过程中的记录,方便自己回顾,好记性不如烂笔头。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值