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)