反射

反射

当一个字节码文件加载到内存的时候,jvm会对该字节码进行解剖,然后会创建一个对象的Class对象,把字节码文件的信息全部都存储到该Class对象中,只要获取到Class对象,就可以使用字节码对象设置对象的属性或者调用对象的方法等操作

类字节码文件是在硬盘上存储的,是一个个的.class文件。在new一个对象时,JVM会先把字节码文件的信息读出来放到内存中,第二次用时,就不用在加载了,而是直接使用之前缓存的这个字节码信息。

字节码的信息包括:类名、声明的方法、声明的字段等信息。在Java中“万物皆对象”,这些信息当然也需要封装一个对象,这就是Class类、Method类、Field类。

通过Class类、Method类、Field类等等类可以得到这个类型的一些信息,甚至可以不用new关键字就创建一个实例,可以执行一个对象中的方法,设置或获取字段的值,这就是反射技术。

注意: 在反射技术中一个类的任何成员都有对应的类进行描述。 比如: 成员变量(Field) 方法----> Method类

一、Class类

1.获取Class对象的三种方式

Java中有一个Class类用于代表某一个类的字节码。

Java提供了三种方式获取类的字节码:

  • 通过ClassLoader对象的loadClass()方法
  • Class.forName():forName方法用于加载某个类的字节码到内存中,并使用class对象进行封装
  • 类名.class
  • 对象.getClass()

Demo1:

public class Demo1 {
    
    public static void main(String[] args) throws ClassNotFoundException {
        //Person p = new Person(123,"hcx");
        
        //推荐使用: 获取Class对象的方式一
        Class clazz1 = Class.forName("com.hcx.reflect.Person");
        System.out.println("clazz1:"+ clazz1);
        
        
        //获取Class对象的方式二: 通过类名获取
        Class clazz2 = Person.class;
        System.out.println("clazz1==clazz2?"+ (clazz1==clazz2));//true
        
        
        //获取Class对象的方式三 :通过对象获取
        Class clazz3 = new Person(123,"hcx").getClass();
        System.out.println("clazz2==clazz3?"+ (clazz2==clazz3));//true
        
    }
    
}

Person:

public class Person {
    
    private int id;
    
    String name;
    
    public Person(int id,String name){
        this.id = id;
        this.name = name;
    }
    
    public Person(){}
    
    
    public void eat(int num){
        System.out.println(name+"吃"+num+"碗饭");
    }
    
    private static void sleep(int num){
        System.out.println("每天睡"+num+"小时");
    }
    
    
    public static void  sum(int[] arr){
        System.out.println("长度是:"+ arr.length);
    }
    
    
    @Override
    public String toString() {
        return " 编号:"+ this.id +" 姓名:"+ this.name;
    }
    
}

2.通过Class类获取类型的一些信息

  1. getName()类的名称(全名,全限定名)
  2. getSimpleName()类的的简单名称(不带包名)
  3. getModifiers(); 类的的修饰符
  4. 创建对象
    无参数构造创建对象
    newInstance()
  5. 获取指定参数的构造器对象,并可以使用Constructor对象创建一个实例
    Constructor<T> getConstructor(Class<?>... parameterTypes)

通过Class对象获取构造方法:

public class Demo2 {
    
    public static void main(String[] args) throws Exception {
        //获取到对应的Class对象
        Class clazz = Class.forName("cn.itcast.reflect.Person");
        
        //通过Class对象获取对应的构造方法
        /*Constructor[] constructors = clazz.getConstructors();  // getConstructors()获取一个类的所有公共的构造方法
        for(Constructor constructor : constructors){
            System.out.println(constructor);
        }
        
        Constructor[] constructors =  clazz.getDeclaredConstructors(); //获取到一个类的所有构造方法,包括私有的在内 。
        for(Constructor constructor : constructors){
            System.out.println(constructor);
        }
        */
        
        /*Constructor constructor = clazz.getConstructor(int.class,String.class);  // getConstructor 获取单个指定的构造方法。
        Person p  = (Person) constructor.newInstance(123,"小红"); // newInstance()创建一个对象
        System.out.println(p);*/
        
        
        
        //获取私有的构造函数
        Constructor constructor =  clazz.getDeclaredConstructor(null);
        //暴力反射
        constructor.setAccessible(true);
        Person p  = (Person) constructor.newInstance(null);
        System.out.println(p);

    }

}

3.通过Class类获取类型中的方法的信息

在反射技术中使用Method类描述方法。

1.获取公共方法包括继承的父类的方法
getMethods()返回一个数组,元素类型是Method
2.获取指定参数的公共方法
getMethod("setName", String.class);
3.获得所有的方法,包括私有
Method[] getDeclaredMethods()
4.获得指定参数的方法,包括私有
Method getDeclaredMethod(String name, Class<?>... parameterTypes)

通过Class对象获取到对应的方法:

public class Demo3 {
    
    public static void main(String[] args) throws Exception {
        //获取到对应的Class对象
        Class clazz = Class.forName("com.hcx.reflect.Person");
        //获取到所有公共的方法
        /*Method[] methods = clazz.getMethods(); // getMethods() 获取所有 的公共方法而已。
        Method[] methods = clazz.getDeclaredMethods(); //获取到所有的方法,但是不包含父类的方法。
        for(Method method  : methods){
            System.out.println(method);
        }*/
        
        Person p = new Person(123,"hcx");
        /*  
        Method m = clazz.getMethod("eat", int.class);
        m.invoke(p, 3); //invoke 执行一个方法。 第一个参数:方法的调用对象。 第二参数: 方法所需要的参数。
        
        
        //执行私有的方法
        Method m =clazz.getDeclaredMethod("sleep",int.class);
        //设置访问权限允许访问
        m.setAccessible(true);
        m.invoke(null, 6);*/
        
        Method m = clazz.getMethod("sum", int[].class);
        m.invoke(p,new int[]{1,2,3});
        
    }
    

}

4.通过Class类获取类型中的字段的信息

在反射技术中使用Field类描述成员变量。

1.获取公共字段
Field[] getFields()
2.获取指定参数的公共字段
Field getField(String name)
3.获取所有的字段
Field[] getDeclaredFields()
4.获取指定参数的字段,包括私用
Field getDeclaredField(String name)

通过反射获取对应的成员变量:

public class Demo4 {

    public static void main(String[] args) throws Exception {
        //获取到对应的Class对象
        Class clazz = Class.forName("com.hcx.reflect.Person");
        //获取 到所有的成员变量
        /*Field[] fields = clazz.getDeclaredFields();
        for(Field field  : fields){
            System.out.println(field);
        }*/
        Person p = new Person();
        Field field = clazz.getDeclaredField("id");
        //设置访问权限可以访问
        field.setAccessible(true);
        field.set(p, 110); //第一个参数: 设置该数据的成员变量,第二个参数:属性值。
        System.out.println(p);
    }
}

二、优缺点

优点:

  • 增加程序的灵活性,避免将固有的逻辑程序写死到代码里
  • 代码简洁,可读性强,可提高代码的复用率

缺点

  • 相较直接调用在创建对象比较多的情景下反射性能下降
  • 内部暴露和安全隐患(破坏单例)

反射性能慢原因

  • 寻找类Class字节码的过程,比如通过ClassName找到对应的字节码Class,然后进行加载、解析,也会比较慢,而new的方式则无需寻找,因为在Linking的解析阶段已经将符号引用转为了直接引用
  • 安全管理机制的权限验证等等
  • 若需要调用native方法调用时JNI接口的使用
  • 入参校验

破坏单例:

public class SingleDemo {

    private static SingleDemo instance; //

    private SingleDemo(){
        if (instance!=null){
            throw  new RuntimeException("这是一个单例,不允许重复创建");
        }
    }

    public static  SingleDemo getInstance(){
        if(instance == null){
            instance = new SingleDemo();
        }
        return instance;
    }
}
public class SingleMain {

    public static void main(String[] args) throws Exception {

        SingleDemo instance = SingleDemo.getInstance();
        SingleDemo instance1 = SingleDemo.getInstance();
        System.out.println(instance == instance1);  //true

        //通过反射获取
        Class<? extends SingleDemo> clazz = instance.getClass();
        Constructor<? extends SingleDemo> constructor = clazz.getDeclaredConstructor();
        constructor.setAccessible(true);//破会访问限制
        SingleDemo instance2 = constructor.newInstance(); //private
        SingleDemo instance3 = clazz.newInstance(); //constructor.newInstance()

        System.out.println(instance==instance2);//false


    }
}

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,457评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,837评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,696评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,183评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,057评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,105评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,520评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,211评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,482评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,574评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,353评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,213评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,576评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,897评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,174评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,489评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,683评论 2 335

推荐阅读更多精彩内容