Java设计模式-单例模式

 

(尊重劳动成果,转载请注明出处:https://blog.csdn.net/qq_25827845/article/details/52422098冷血之心的博客)

目录

单例模式

饿汉式:

懒汉式:

 结论:

2017-08-09更新全部的单例模式的写法,代码如下:

2018-12-01日,新增了使用内部静态枚举Enum来实现单例模式


单例模式

单例模式是一种常用的软件设计模式。在它的核心结构中只包含一个被称为单例的特殊类。通过单例模式可以保证系统中一个类只有一个实例而且该实例易于外界访问,从而方便对实例个数的控制并节约系统资源。如果希望在系统中某个类的对象只能存在一个,单例模式是最好的解决方案。

      单例模式分为俩种,即懒汉式(实例延迟加载模式)和饿汉式。分别举例如下:

饿汉式:

/*
//饿汉式。
/*
class Single
{
	private static final Single s = new Single();
	private Single(){}
	public static Single getInstance()
	{
		return s;
	}
}

饿汉式没有问题,话不多说。

懒汉式:

懒汉式实例延迟加载的单例模式

最简单的懒汉式如下:

class Single
{
	private static Single s = null;
	private Single(){}

	public static  Single getInstance()
	{
		if(s==null)
		   s = new Single();
		return s;
	}
}

然而其问题比较多,单线程执行时可以,多线程时会出现线程安全问题。

解决办法------使用同步代码块,使线程同步。通过增加判断次数来实现。

//懒汉式

class Single
{
	private static Single s = null;
	private Single(){}


	public static  Single getInstance()
	{
		if(s==null)
		{
			synchronized(Single.class)
			{
				if(s==null)
					//--->A;
					s = new Single();
			}
		}
		return s;
	}
}

class SingleDemo 
{
	public static void main(String[] args) 
	{
		System.out.println("Hello World!");
	}
}

       此例子中,线程1调用getInstance()方法,经过判断,进入到同步代码块中,并且上锁。此时线程1还没来得及创建对象。所以线程2便进入了第一个if语句块,但是被锁在了同步块外边。当线程1创建了对象后,并且退出同步代码块。线程2进入之后也不会再次创建对象。后边的线程将不会进入if语句块,所以减少了同步锁的判断次数。而是直接return s;提高了懒汉式的执行效率。

 结论:

通过双重判断来提高了懒汉式单例模式的执行效率。

2017-08-09更新全部的单例模式的写法,代码如下:

/**
 * 单线程环境下的饿汉式单例模式    
 * @author ywq
 *
 *
 */
class Single{  
    private static final Single s = new Single();  
    private Single(){}  
    public static Single getInstance(){  
        return s;  
    }  
} 

/**
 * 单线程环境下的懒汉式单例模式    
 * @author ywq
 *
 *
 */
class Single{  
    private static Single s = null;  
    private Single(){}  
  
    public static  Single getInstance(){  
        if(null==s)  
           s = new Single();  
        return s;  
    }  
} 

/**
 * 多线程环境下的懒汉式单例模式(简单加锁)    
 * @author ywq
 *
 *
 */
class Single{  
    private static Single s = null;  
    private Single(){}  
  
    public static  Single getInstance(){
    	synchronized(Single.class){
    		if(null==s)  
    			s = new Single();  
    	}
    	return s;  
    }  
} 

/**
 * 多线程环境下的懒汉式单例模式(DCL,双检锁实现) 
 * 由于指令可能重排序,即DCL可能会返回一个并不完整的对象。   
 * @author ywq
 *
 *
 */
class Single{  
    private static Single s = null;  
    private Single(){}  
  
    public static  Single getInstance(){
    	if(null==s){
    		synchronized(Single.class){
        		if(null==s)  
        			s = new Single();  
        	}
    	}
    	return s;  
    }  
}


/**
 * 多线程环境下的懒汉式单例模式(DCL,双检锁+volatile实现) 
 * 加入了volatile变量来禁止指令重排序   
 * @author ywq
 *
 *
 */
class Single{  
    private static volatile Single s = null;  
    private Single(){}  
  
    public static  Single getInstance(){
    	if(null==s){
    		synchronized(Single.class){
        		if(null==s)  
        			s = new Single();  
        	}
    	}
    	return s;  
    }  
}

/**
 * 多线程环境下的懒汉式单例模式(基于静态内部类实现)    
 * @author ywq
 *
 *
 */
class Single{  
	
    private Single(){}  
    private static class Inner{
    	final static Single s = new Single();
    }
    public static  Single getInstance(){
    	return Inner.s;
    }  
}

2018-12-01日,新增了使用内部静态枚举Enum来实现单例模式

/**
 * 使用枚举的单例模式
 */
public class EnumSingleton{
    private EnumSingleton(){}
    public static EnumSingleton getInstance(){
        return Singleton.INSTANCE.getInstance();
    }
    
    private static enum Singleton{
        INSTANCE;
        
        private EnumSingleton singleton;
        //JVM会保证此方法绝对只调用一次
        private Singleton(){
            singleton = new EnumSingleton();
        }
        public EnumSingleton getInstance(){
            return singleton;
        }
    }
}

 

如果对你有帮助,记得点赞哦~欢迎大家关注我的博客,我会持续更新后续章节学习笔记,可以进群366533258一起交流学习哦~

本群给大家提供一个学习交流的平台,内设菜鸟Java管理员一枚、精通算法的金牌讲师一枚、Android管理员一枚、蓝牙BlueTooth管理员一枚、Web前端管理一枚以及C#管理一枚。欢迎大家进来交流技术。

 

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

温柔狠角色

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值