• 终结者单身——setAccessible(true)


    首先看一下“传说”Singleton模式

    package go.derek;
    
    public class Singleton{
    	
    	public static int times;
    	private Singleton(){
    		//构造器被调用的时候会打印出次数
    		System.out.println("单例构造器被调用"+(++times)+"两次");
    	}
    	
    	private final static Singleton instance=new Singleton();
    	
    	public static Singleton getInstance(){
    		return instance;
    	}
    	
    	public void doSomething(){
    		System.out.println("do something");
    	}
    	
    }
    
    以下是測试类主函数:
    package go.derek;
    
    import java.lang.reflect.Constructor;
    import java.lang.reflect.InvocationTargetException;
    
    
    public class Test {
    	public static void main(String[] args) throws NoSuchMethodException, SecurityException, InstantiationException, IllegalAccessException, IllegalArgumentException, InvocationTargetException {
    		//通过单例模式获得单例对象obj1
    		Singleton obj1=Singleton.getInstance();
    		//运行一次doSomething方法
    		obj1.doSomething();
    		//观察控制台,这次获得的obj2对象跟obj1是同一个单例。没有调用构造器
    		Singleton obj2=Singleton.getInstance();
    		obj2.doSomething();
    		//以下厉害的来了,首先拿到万能的Class对象
    		Class<Singleton> clazz=Singleton.class;
    		//然后拿到构造器。使用这种方法私有的构造器也能够拿到
    		Constructor<Singleton> c=clazz.getDeclaredConstructor();
    		//设置在使用构造器的时候不运行权限检查
    		c.setAccessible(true);
    		//因为没有了权限检查。所以在Singleton类外面也能够创建对象了,然后运行方法
    		//观察控制台,私有构造器又被调用了一次,单例模式被攻陷了,运行方法成功。

    c.newInstance().doSomething(); } }


    执行结果例如以下:

    单例构造器被调用1两次
    do something
    do something
    单例构造器被调用2两次
    do something

    试想一下,假设某个恶意client通过上面的方式。就能够为所欲为了。所以为了避免出现这样的情况,能够再构造器被第二次调用的时候抛出一个异常

    package go.derek;
    
    public class Singleton{
    	
    	public static int times;
    	private Singleton() {
    		//构造器被调用的时候会打印出次数
    		System.out.println("单例构造器被调用"+(++times)+"两次");
    		if(instance!=null){
    			throw new IllegalArgumentException("单例构造器不能反复使用");
    		}
    	}
    	
    	private final static Singleton instance=new Singleton();
    	
    	public static Singleton getInstance(){
    		return instance;
    	}
    	
    	public void doSomething(){
    		System.out.println("do something");
    	}
    	
    }
    
    执行结果例如以下:

    单例构造器被调用1两次
    do something
    do something
    单例构造器被调用2两次
    Exception in thread "main" java.lang.reflect.InvocationTargetException
    at sun.reflect.NativeConstructorAccessorImpl.newInstance0(Native Method)
    at sun.reflect.NativeConstructorAccessorImpl.newInstance(NativeConstructorAccessorImpl.java:57)
    at sun.reflect.DelegatingConstructorAccessorImpl.newInstance(DelegatingConstructorAccessorImpl.java:45)
    at java.lang.reflect.Constructor.newInstance(Constructor.java:526)
    at go.derek.Test.main(Test.java:24)
    Caused by: java.lang.IllegalArgumentException: 单例构造器不能反复使用
    at go.derek.Singleton.<init>(Singleton.java:10)
    ... 5 more

    目标实现~

    版权声明:本文博客原创文章,博客,未经同意,不得转载。

  • 相关阅读:
    IDEA常用快捷指令整理
    Python dict 字典
    内联函数
    【MFC】编辑框 CEdit Ctrl控件自动换行设置
    mysql 多sql文件恢复方案
    Linux: 用64位的系统,能编译32位的程序吗?
    C++ 遍历数组
    工业现场传感器传感器为什么采用电流形式输出?
    【转】电磁阀、电磁铁的工作原理说明
    PCB板强弱电隔离距离不够导致损坏和问题检查记录
  • 原文地址:https://www.cnblogs.com/lcchuguo/p/4752033.html
Copyright © 2020-2023  润新知