Java反射机制

Java中“一切”都是对象。就算对象的模板,即类本身也可以用对象来表示。Java反射机制就是来操作类对象的。

一.获取Class对象

首先来看一下如何获取到类对象,即java.lang.Class对象。
1.使用getClass方法
Object类有个getClass实例方法。而其它类都派生于Object类。所以任何对象都可以调用getClass方法来获取表示自己类对象。

System.out.println("abc".getClass().getName()); //输出:java.lang.String

2.使用Class.forName方法
Class类本身带有一个静态方法:forName,接收一个表示类的全限定名的字符串参数,返回一个该类的类对象,如果找不到则抛出异常。

 String name = Class.forName("java.lang.String").getName();
 System.out.println(name); //输出:java.lang.String

3.类名.class
如String.class、int.class、Double[].class

二.利用反射分析类的能力

java.lang.reflect包中有三个类Field、Method和Constructor分别用于描述类的域、方法和构造方法。

类似的方法

  • 都有一个getName方法,用来返回名字;
  • 都有getModifiers方法,返回一个整形值,描述修饰符,可以使用Modifier.toString(int)方法来得到字符串形式的修饰符;

Field

  • getType方法,可以返回属性类型的Class对象;

Method

  • getParameterTypes方法,可以返回形参的Class对象数组;
  • getReturnType方法可以得到返回值类型的Class对象.

Constructor

  • 有个getParameterTypes方法,可以返回形参的Class对象数组;

Class

  • static forName(String),根据全限定名找到对应的Class对象,可能抛出ClassNotFoundException
  • getSuperClass(),得到父类的Class对象
  • getFields() 返回public属性
  • getMethods()返回public方法
  • getConstructors()返回public构造方法
  • getDeclareFields()返回全部属性
  • getDeclareMethods()返回全部方法
  • getDeclareConstructors()返回全部构造方法

代码例子

三.在运行时使用反射分析对象

运行时,查看对象属性、设置对象属性。
先使用getDeclaredField(String)获得想要的Field,或者getDeclaredField()获取Field[]数组。
使用Field.get(Object obj),obj是要查询的对象的,获取到属性值。如果是int等基本数据类型,可以使用getInt(Object obj)等获取。
使用Field.set(Object obj,Object value),设置obj的相应属性为value。
如果当前代码对于目标对象的属性没有访问权限,get,set等方法会抛出非法参数异常。可以使用Field.setAccessible(true)来取消访问权限限制

public class ObjectAnalyzer {
    public static void main(String[] args) throws Exception {
        Class clazz = Class.forName("cn.xiaozhigang.code.Basic.BasicGramer.reflex.ReflexTest");
        Field fieldName = clazz.getDeclaredField("name");
        //因为是私有属性,需要设置访问权限
        fieldName.setAccessible(true);

        ReflexTest reflexTest = new ReflexTest();
        System.out.println(fieldName.get(reflexTest));
        //设置对象reflexTest的name属性为hello
        fieldName.set(reflexTest, "hello");
        System.out.println(reflexTest);

        Field fieldAge = clazz.getDeclaredField("age");
        fieldAge.setAccessible(true);
        //注意目标类的age属性为int类型,是基本数据类型。而Filed.get方法返回的是Object
        //所以这里用了getInt方法来得到其值
        int age = fieldAge.getInt(reflexTest);
        System.out.println(age);
        fieldAge.set(reflexTest,11);
        System.out.println(reflexTest);
    }
}

程序输出:

test
ReflexTest{name='hello', age=10}
10
ReflexTest{name='hello', age=11}

四.调用任意方法

Method类有一个invoke方法,运行执行任意对象的方法。invoke方法的形参包括,要执行方法的对象和参数。其方法声明为:Object invoke(Object obj,Object...args)

获取到指定的Method对象:
首先要获取到类的Class对象。再调用Class对象的getMethod("方法名")方法。如果方法形参,则还应该跟上方法的形参类型的Class对象:

Method m1 = TestObj.class.getMethod("getName");
Method m2 = TestObj.class.getMethod("raiseSalary",double.class)

你可能感兴趣的:(Java反射机制)