JAVA设计模式:单例模式

Java中单例模式是一种常见的设计模式,单例模式的写法有好几种,这里主要介绍二种:懒汉式单例、饿汉式单例。

单例模式Singleton
 * 应用场合:有些对象只有一个就够了,如皇帝和老婆

 1. Windows的Task Manager(任务管理器)

   2. windows的Recycle Bin(回收站)

   3. 数据库连接池的设计一般也是采用单例模式

   ..............

 * 作用:保证整个应用程序中,某个实例有且只有一个

 

饿汉模式

public class Singleton {
	//饿汉模式->在类加载的时候,它的实例就加载了(private static Singleton instance=new Singleton())
	//将构造方法私有化,不允许外界直接创造对象
	private Singleton(){
		
	}
	//在类的内部创建一个唯一实例
	private static Singleton instance=new Singleton();
	
	//提供一个获取实例的方法
	public static Singleton getInstance(){
		return instance;
	}
	
}

 

懒汉模式

public class Singleton2 {
	//1.将构造方法私有化,不允许外部直接创建对象
	private Singleton2(){
		
	}
	//2.声明类的唯一实例,使用private static修饰
	private static Singleton2 instance;
	
	//3.提供一个获取实例的方法,使用public static修饰
	public static Singleton2 getInstance(){
		if(instance==null){
			instance=new Singleton2();
		}
		return instance;
	}
}

测试:

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

结果是

"s1和s2是同一个实例“

”s3和s4是同一个实例“

但是以上懒汉模式的实现没有考虑线程安全的问题,在并发情况下可能出现多个Singleton实例,可以加上synchronized,不过最好加在getInstance方法里面

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


synchronized 写在方法上,为什么效率低?
在N个线程同时都在getInstance的情况下,如果synchronized 加在方法上,那么每次getInstance都需要加锁,解锁。
意味着:获取该类的实例对象时,N个线程都必须排队,一个个等着获取锁。

 

 

为什么synchronized写在里面,要判断两次instance是否为空?

假设程序刚启动,同时100个线程,都在拿instance,这时候,instance还没实例化,运行在最前面的5个线程A,B,C,D,E,因为instance == null
他们都运行到了的if (instance == null)大括号内,同步代码块前。ABCDE,5个线程中的一个“幸运儿D",拿到了锁内代码的执行权(cpu给它分配了时间片),而其他4线程,A,B,C,E个只能在同步代码块外面等候,“幸运儿D",进入代码块后,再次判断instance==null?
发现instance确实没被创建,为null,于是new操作创建instance,拿到了instance,返回,并且把“锁”释放。
剩下的A,B,C,E,也一个一个进入同步代码块,与D不同的是,他们在锁内的判断中都发现instance不等于null,
(很显然同步代码快里面如果没有再次判断,instance会被创建5次...)
不需要new操作,直接拿。


至于剩下的95个线程,他们在instance被创建之后,
才走到第一个判断 if (instance == null) 这里,
这时候,他们发现instance不等于null,已经被创建了
他们都不需要经过同步代码快,所以效率高。
 

======================以下为2019/3/28更新内容===============================

通过枚举实现单例模式,可以有效防范反射、反序列化攻击,这也是effective java这本书中强力推荐的!

public enum  ValidateCodeType {

    /**
     * 短信验证码
     */
    SMS {
        @Override
        public String getParamNameOnValidate() {
            return SecurityConstants.DEFAULT_PARAMETER_NAME_CODE_SMS;
        }
    },
    /**
     * 图片验证码
     */
    IMAGE {
        @Override
        public String getParamNameOnValidate() {
            return SecurityConstants.DEFAULT_PARAMETER_NAME_CODE_IMAGE;
        }
    };

    /**
     * 校验时从请求中获取的参数的名字
     * @return
     */
    public abstract String getParamNameOnValidate();

}

上面是一个简单的枚举类,它有两个变量一个是图片验证码,一个是短信验证码

相信大部分人平时都是很简单的应用enum,这样的写法可能很多人还没见过,现在我们用户jad反编译工具查看下反编译结果!

public abstract class ValidateCodeType extends Enum
{

    public static ValidateCodeType[] values()
    {
        return (ValidateCodeType[])$VALUES.clone();
    }

    public static ValidateCodeType valueOf(String name)
    {
        return (ValidateCodeType)Enum.valueOf(com/wlj/security/core/validate/code/ValidateCodeType, name);
    }

    private ValidateCodeType(String s, int i)
    {
        super(s, i);
    }

    public abstract String getParamNameOnValidate();


    public static final ValidateCodeType SMS;
    public static final ValidateCodeType IMAGE;
    private static final ValidateCodeType $VALUES[];

    static
    {
        SMS = new ValidateCodeType("SMS", 0) {

            public String getParamNameOnValidate()
            {
                return "smsCode";
            }

        }
;
        IMAGE = new ValidateCodeType("IMAGE", 1) {

            public String getParamNameOnValidate()
            {
                return "imageCode";
            }

        }
;
        $VALUES = (new ValidateCodeType[] {
            SMS, IMAGE
        });
    }
}

ValidateCodeType是一个抽象类继承了Enum,只有一个私有构造函数(不能在外部被调用),这符合单例模式的定义,另外还定义一个抽象方法以及两个ValidateCodeType类型的静态final修饰的变量

这两个变量在static代码块中被初始化(饿汉模式),赋值匿名内部类实例的引用(使用匿名内部类我们必须要继承一个父类或者实现一个接口,当然也仅能只继承一个父类或者实现一个接口。同时它也是没有class关键字,这是因为匿名内部类是直接使用new来生成一个对象的引用。当然这个引用是隐式的)

既然是继承了外部的抽象类,就要实现它的抽象方法,

 

 

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值