Java 反射 (Java Reflection)

JAVA反射机制

编辑JAVA反射机制是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意方法和属性;这种动态获取信息以及动态调用对象方法的功能称为java语言的反射机制。

JAVA反射(放射)机制:

“程序运行时,允许改变程序结构或变量类型,这种语言称为动态语言”。

优点

灵活性高。因为反射属于动态编译,即只有到运行时才动态创建 &获取对象实例。
编译方式说明:

  1. 静态编译:在编译时确定类型 & 绑定对象。如常见的使用new关键字创建对象
  2. 动态编译:运行时确定类型 & 绑定对象。动态编译体现了Java的灵活性、多态特性 & 降低类之间的藕合性

缺点

执行效率低
因为反射的操作 主要通过JVM执行,所以时间成本会 高于 直接执行相同操作

  1. 因为接口的通用性,Java的invoke方法是传object和object[]数组的。基本类型参数需要装箱和拆箱,产生大量额外的对象和内存开销,频繁促发GC。
  2. 编译器难以对动态调用的代码提前做优化,比如方法内联。
  3. 反射需要按名检索类和方法,有一定的时间开销.
  4. 容易破坏类结构 ,因为反射操作饶过了源码,容易干扰类原有的内部逻辑

应用场景

动态获取 类文件结构信息(如变量、方法等) & 调用对象的方法
常用的需求场景有:动态代理、工厂模式优化、Java JDBC数据库操作等

Java反射机制提供的功能

image.png

java.lang.Class 类

定义:java.lang.Class类是反射机制的基础
作用:存放着对应类型对象的 运行时信息

  1. 在Java程序运行时,Java虚拟机为所有类型维护一个java.lang.Class对象
  2. 该Class对象存放着所有关于该对象的 运行时信息
  3. 泛型形式为Class<T>

每种类型的Class对象只有1个 = 地址只有1个

        // 对于2个String类型对象,它们的Class对象相同
        Class c1 = "class".getClass();
        Class c2 = Class.forName("java.lang.String");
        // 用==运算符实现两个类对象地址的比较
        System.out.println(c1 == c2);
        // 输出结果:true

实现方式

反射机制的实现 主要通过 操作java.lang.Class类

  1. 获取 目标类型的Class对象
  2. 通过 Class 对象分别获取Constructor类对象、Method类对象 & Field 类对象
  3. 通过 Constructor类对象、Method类对象 & Field类对象分别获取类的构造函数、方法&属性的具体信息,并进行后续操作

获取 目标类型的Class对象的方式主要有4种方法

    private static void getTargetClass() throws ClassNotFoundException {
        /** 方式1:Object.getClass()
         * Object类中的getClass()返回一个Class类型的实例*/
        Boolean temp = true;
        Class type = temp.getClass();
        System.out.println(type);//结果:class java.lang.Boolean

        /** 方式2:T.class 语法,T = 任意Java类型
         * 注:Class对象表示的是一个类型,而这个类型未必一定是类
         *如,int不是类,但int.class是一个Class类型的对象*/
        Class temp2 = Boolean.class;
        System.out.println(temp2);//结果:class java.lang.Boolean
        Class temp21 = int.class;
        System.out.println(temp21);//结果:int


        /** 方式3:static method Class.forName
         Class<?> classType = Class . forName ("java.lang.Boolean"); */
        Class temp3 = Class.forName("java.lang.Boolean");
        System.out.println(temp3);//结果:class java.lang.Boolean

        /** 方式4:TYPE语法
         * Class<?> classType = Boolean . TYPE*/
        Class<?> temp4 = Boolean.TYPE;
        System.out.println(temp4);//结果:boolean
    }

反射类对应关系

Java反射机制的实现除了依靠Java.lang.Class类,还需要依靠:Constructor类、Field类、Method类,分别作用于类的各个组成部分:


image.png

Type类

Type是Java中所有类型的通用超级接口编程语言。这些包括原始类型,参数化类型,数组类型、类型变量和原始类型
以下为继承Type子类

image.png

通过 Class 对象分别获取Constructor类对象、Method类对象 & Field 类对

// 即以下方法都属于`Class` 类的方法。

<-- 1. 获取类的构造函数(传入构造函数的参数类型)->>
  // a. 获取指定的构造函数 (公共 / 继承)
  Constructor<T> getConstructor(Class<?>... parameterTypes)
  // b. 获取所有的构造函数(公共 / 继承) 
  Constructor<?>[] getConstructors(); 
  // c. 获取指定的构造函数 ( 不包括继承)
  Constructor<T> getDeclaredConstructor(Class<?>... parameterTypes) 
  // d. 获取所有的构造函数( 不包括继承)
  Constructor<?>[] getDeclaredConstructors(); 
// 最终都是获得一个Constructor类对象

// 特别注意:
  // 1. 不带 "Declared"的方法支持取出包括继承、公有(Public) & 不包括有(Private)的构造函数
  // 2. 带 "Declared"的方法是支持取出包括公共(Public)、保护(Protected)、默认(包)访问和私有(Private)的构造方法,但不包括继承的构造函数
  // 下面同理

<--  2. 获取类的属性(传入属性名) -->
  // a. 获取指定的属性(公共 / 继承)
   Field getField(String name) ;
  // b. 获取所有的属性(公共 / 继承)
   Field[] getFields() ;
  // c. 获取指定的所有属性 (不包括继承)
   Field getDeclaredField(String name) ;
  // d. 获取所有的所有属性 (不包括继承)
   Field[] getDeclaredFields() ;
// 最终都是获得一个Field类对象

<-- 3. 获取类的方法(传入方法名 & 参数类型)-->
  // a. 获取指定的方法(公共 / 继承)
    Method getMethod(String name, Class<?>... parameterTypes) ;
  // b. 获取所有的方法(公共 / 继承)
   Method[] getMethods() ;
  // c. 获取指定的方法 ( 不包括继承)
   Method getDeclaredMethod(String name, Class<?>... parameterTypes) ;
  // d. 获取所有的方法( 不包括继承)
   Method[] getDeclaredMethods() ;
// 最终都是获得一个Method类对象

<-- 4. Class类的其他常用方法 -->
getSuperclass(); 
// 返回父类

String getName(); 
// 作用:返回完整的类名(含包名,如java.lang.String ) 

Object newInstance(); 
// 作用:快速地创建一个类的实例
// 具体过程:调用默认构造器(若该类无默认构造器,则抛出异常 
// 注:若需要为构造器提供参数需使用java.lang.reflect.Constructor中的newInstance()

通过 Constructor类对象、Method类对象 & Field类对象分别获取类的构造函数方法 & 属性的具体信息 & 进行操作

// 即以下方法都分别属于`Constructor`类、`Method`类 & `Field`类的方法。

<-- 1. 通过Constructor 类对象获取类构造函数信息 -->
  String getName();// 获取构造器名
  Class getDeclaringClass();// 获取一个用于描述类中定义的构造器的Class对象
  int getModifiers();// 返回整型数值,用不同的位开关描述访问修饰符的使用状况
  Class[] getExceptionTypes();// 获取描述方法抛出的异常类型的Class对象数组
  Class[] getParameterTypes();// 获取一个用于描述参数类型的Class对象数组

<-- 2. 通过Field类对象获取类属性信息 -->
  String getName();// 返回属性的名称
  Class getDeclaringClass(); // 获取属性类型的Class类型对象
  Class getType();// 获取属性类型的Class类型对象
  int getModifiers(); // 返回整型数值,用不同的位开关描述访问修饰符的使用状况
  Object get(Object obj) ;// 返回指定对象上 此属性的值
  void set(Object obj, Object value) // 设置 指定对象上此属性的值为value

<-- 3. 通过Method 类对象获取类方法信息 -->
  String getName();// 获取方法名
  Class getDeclaringClass();// 获取方法的Class对象 
  int getModifiers();// 返回整型数值,用不同的位开关描述访问修饰符的使用状况
  Class[] getExceptionTypes();// 获取用于描述方法抛出的异常类型的Class对象数组
  Class[] getParameterTypes();// 获取一个用于描述参数类型的Class对象数组

<--额外:java.lang.reflect.Modifier类 -->
// 作用:获取访问修饰符

static String toString(int modifiers)   
// 获取对应modifiers位设置的修饰符的字符串表示

static boolean isXXX(int modifiers) 
// 检测方法名中对应的修饰符在modifiers中的值

举个例子,获取String 所有构造方法

image.png

特别注意:访问权限问题

反射机制的默认行为受限于Java的访问控制
如,无法访问( private )私有的方法、字段

冲突

Java安全机制只允许查看任意对象有哪些域,而不允许读它们的值
若强制读取,将抛出异常

解决方案

脱离Java程序中安全管理器的控制、屏蔽Java语言的访问检查,从而脱离访问控制
具体实现手段:使用Field类、Method类 & Constructor类对象的setAccessible()

void setAccessible(boolean flag)    
// 作用:为反射对象设置可访问标志
// 规则:flag = true时 ,表示已屏蔽Java语言的访问检查,使得可以访问 & 修改对象的私有属性

boolean isAccessible()  
// 返回反射对象的可访问标志的值

static void setAccessible(AccessibleObject[] array, boolean flag)   
// 设置对象数组可访问标志

实例1:利用反射获取类的属性 & 赋值

 /**
     * 测试类定义
     */
    public static class Student {
        private String name;

        public Student() {
            System.out.println("创建了一个Student实例");
        }
    }

    /**
     * 测试方法
     */
    private static void example() throws Exception {
        //利用反射获取属性 & 赋值
        // 1. 获取Student类的Class对象
        Class<Student> studentClass = Student.class;

        // 2. 通过Class对象创建Student类的对象
        Constructor<?> constructor = studentClass.getDeclaredConstructor();
        constructor.setAccessible(true);
        Object mStudent = studentClass.newInstance();

        // 3. 通过Class对象获取Student类的name属性
        Field f = studentClass.getDeclaredField("name");

        // 4. 设置私有访问权限
        f.setAccessible(true);

        // 5. 对新创建的Student对象设置name值
        f.set(mStudent, "我是java 反射");

        // 6. 获取新创建Student对象的的name属性 & 输出
        System.out.println(f.get(mStudent));
    }

测试结果


image.png

利用反射调用类的构造函数

/**
     * 测试类定义
     */
    public static class Student2 {
        private String name;

        public Student2() {
            System.out.println("无参构造 ");
        }

        public Student2(String str) {
            System.out.println("有参构造 " + "****" + str);
        }
    }

    /**
     * 利用反射调用构造函数
     */
    private static void example2() throws Exception {
        //利用反射调用构造函数
        // 1. 获取Student类的Class对象
        Class studentClass = Student2.class;

        // 2.1 通过Class对象获取Constructor类对象,从而调用无参构造方法
        // 注:构造函数的调用实际上是在newInstance(),而不是在getConstructor()中调用
        Object mObj1 = studentClass.getConstructor().newInstance();

        // 2.2 通过Class对象获取Constructor类对象(传入参数类型),从而调用有参构造方法
        Object mObj2 = studentClass.getConstructor(String.class).newInstance("OK");
    }

[图片上传失败...(image-1c62c9-1532574152311)]

利用反射调用方法

/**
     * 测试类定义
     */
    public static class Student {
        public Student() {
            System.out.println("创建了一个Student实例");
        }

        // 无参数方法
        public void setName1() {
            System.out.println("调用了无参方法:setName1()");
        }

        // 有参数方法
        public void setName2(String str) {
            System.out.println("调用了有参方法setName2(String str):" + str);
        }
    }

    /**
     * 利用反射调用方法
     */
    private static void example2() throws Exception {
        //利用反射调用方法
        // 1. 获取Student类的Class对象
        Class studentClass = Student.class;

        // 2. 通过Class对象创建Student类的对象
        Object mStudent = studentClass.newInstance();

        // 3.1 通过Class对象获取方法setName1()的Method对象:需传入方法名
        // 因为该方法 = 无参,所以不需要传入参数
        Method msetName1 = studentClass.getMethod("setName1");

        // 通过Method对象调用setName1():需传入创建的实例
        msetName1.invoke(mStudent);

        // 3.2 通过Class对象获取方法setName2()的Method对象:需传入方法名 & 参数类型
        Method msetName2 = studentClass.getMethod("setName2", String.class);

        // 通过Method对象调用setName2():需传入创建的实例 & 参数值
        msetName2.invoke(mStudent, "Carson_Ho");
    }
image.png

原文作者博客地址,点击查看

原文地址鸿洋微信公众号,点击查看原文

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

推荐阅读更多精彩内容