Spring 在创建 Bean 实例和依赖注入以及AOP时都使用了反射,今天我们就来讲解一下反射的概念以及其应用。

测试开发必须掌握的知识点:Java反射_反射

反射机制

Java反射是Java被视为动态(或准动态)语言的一个关键性质。这个机制允许程序在运行时通过Reflection APIs取得任何一个已知名称的class的内部信息以及任意一个对象的内部信息。Java反射机制提供如下功能:

在运行时判断任意一个对象所属的类

在运行时构造任意一个类的对象

在运行时判断任意一个类所具有的成员变量和方法

在运行时调用任一个对象的方法

在运行时创建新类对象

在使用Java的反射功能时,基本首先都要获取类的Class对象,再通过Class对象获取其他的对象。大家都知道,在Junit4中注解@Test表示测试用例,每一个测试用例的本质就是测试类中的一个方法,即:

@Test

    public void test() {

        fail("Not yet implemented");

    }

我们知道,通常情况下,调用一个类的方法是,先对类进行实例化,记为obj,然后通过obj.test()的方式调用。在这里我们思考一个问题,Junit4是一个框架,在运行的过程中,框架根本不知道用户定义了多少个测试用例(虽然通过@Test进行了约束),显然框架是在运行的时候才确认了测试用例,并通过某种方式调用了测试用例,这就是反射的本质——在运行时工作!

Class类和Class实例

  • java.lang.Class(大写的Class)的作用是运行时提供或获得某个对象的类型信息,可用于反射。
  • java区分大小写,因此Class和关键字class并不冲突。
  • 当我们创建一个类Dog时,Java会自动生成一个内容是Dog的Class类的对象(虚拟机为每种类型管理一个独一无二的Class对象)。
  • Class类的对象只能由JVM创建,无法通过new来创建。

 常用API介绍

在这里我们重点介绍反射技术中关于获取Class对象,访问字段,调用方法以及调用构造方法的API

1.获取类的Class对象

Class(java.lang.Class) 类的实例表示正在运行的 Java 应用程序中的类和接口。这个Class实例是JVM内部创建的,如果我们查看JDK源码,可以发现Class类的构造方法是private,只有JVM能创建Class实例,我们自己的Java程序是无法创建Class实例的。由于JVM为每个加载的class创建了对应的Class实例,并在实例中保存了该class的所有信息,包括类名、包名、父类、实现的接口、所有方法、字段等,因此,如果获取了某个Class实例,我们就可以通过这个Class实例获取到该实例对应的class的所有信息。获取类的Class对象有多种方式:

测试开发必须掌握的知识点:Java反射_Java_02

2、获取类的Fields

可以通过反射机制得到某个类的某个属性,然后改变对应于这个类的某个实例的该属性值。JAVA 的Class<T>类提供了几个方法获取类的属性。

测试开发必须掌握的知识点:Java反射_反射_03

3.获取类的Method

通过反射机制得到某个类的某个方法,然后调用对应于这个类的某个实例的该方法,Class<T>类提供了几个方法获取类的方法。

测试开发必须掌握的知识点:Java反射_java_04

4.获取类的Constructor

通过反射机制得到某个类的构造器,然后调用该构造器创建该类的一个实例,Class<T>类提供了几个方法获取类的构造器。

测试开发必须掌握的知识点:Java反射_Java_05

反射API应用

写一个类

public class ReflectDemo {

       ReflectDemo(){    

              System.out.println("默认构造函数");

       }

       ReflectDemo(String p_para){      

              System.out.println("有参构造函数");

       }

       public String myPara1="public属性";

       protected String myPara2="protected属性";

       private String myPara3="private属性";

       public void test1(){

              System.out.println("这是 public void 无参方法test1");

       }

       protected String test2(String p_test2){

              System.out.println("这是 protected void 有参方法test2");

              returnp_test2;

       }

       private void test3(){

              System.out.println("这是 privated 无参方法test3");

       }

}

新建类实例

调用类的Class对象的newInstance方法,该方法会调用对象的默认构造器,如果没有默认构造器,会调用失败,代码如下:

Class classType =ReflectDemo.class;

Object inst = classType.newInstance();

System.out.println(inst);

调用默认Constructor对象的newInstance方法,代码如下:

Class classType =ReflectDemo.class;

Constructor constructor1 = classType.getConstructor();

Object inst = constructor1.newInstance();

System.out.println(inst);

调用带参数Constructor对象的newInstance方法,代码如下:

Constructor constructor2 =ReflectDemo.class.getDeclaredConstructor(String.class);

Object inst = constructor2.newInstance("test");

System.out.println(inst);

调用方法

通过反射获取类Method对象,获取类中的所有函数。

String className = "com.lesson.reflect.ReflectDemo";       

Class clas = Class.forName(className);

Method[] a=clas.getDeclaredMethods();

for(int i=0;i<a.length;i++){

       System.out.println(a[i].toString());

}

通过反射获取类Method对象,调用method的Invoke方法调用函数。

调用protected有参方法  ,有参方法

Class simpleClass = Class.forName("com.lesson.reflect.ReflectDemo");

Object simpelObject = simpleClass.newInstance();

Method simpleMethod =simpleClass.getDeclaredMethod("test2",  String.class);

simpleMethod.invoke(simpelObject, "Hello,world");

调用private方法,有参方法

Class simpleClass2 = Class.forName("com.lesson.reflect.ReflectDemo");

Object simpelObject2 = simpleClass2.newInstance();

Method simpleMethod2 = simpleClass2.getDeclaredMethod("test3",   String.class);

simpleMethod2.setAccessible(true);

simpleMethod2.invoke(simpelObject2, "Hello,world");

调用public,无参方法

Class simpleClass3 = Class.forName("com.lesson.reflect.ReflectDemo");

Object simpelObject3 =simpleClass3.newInstance();             

Method simpleMethod3 =simpleClass3.getDeclaredMethod("test1");

simpleMethod3.invoke(simpelObject3);

设置读取属性

通过反射获取类的Field对象,调用Field中的方法设置或获取值

设置或获取private变量 

ReflectDemo t =new ReflectDemo();

Class temp = t.getClass();

Field f;

f = temp.getDeclaredField("myPara3");

f.setAccessible(true); 

System.out.println(f.get(t));

f.set(t, "新的private属性");

System.out.println(f.get(t));

好了,这就是反射的基础API使用方法,可能大家还是不能够理解其在实际工作中的应用价值,接下来我会写一篇文章把java的注解和反射知识点结合起来帮助大家更好的消化,敬请期待!