一、反射的概念 Java反射就是在运行状态中,对于任意一个类,都能够知道这个类的所有属性和方法;对于任意一个对象,都能够调用它的任意方法和属性;并且能改变它的属性。而这也是Java被视为动态(或准动态,为啥要说是准动态,因为一般而言的动态语言定义是程序运行时,允许改变程序结构或变量类型,这种语言称为动态语言。从这个观点看,Perl,Python,Ruby是动态语言,C++,Java,C#不是动态语言。)语言的一个关键性质。
二、反射的作用 我们知道反射机制允许程序在运行时取得任何一个已知名称的class的内部信息,包括包括其modifiers(修饰符),fields(属性),methods(方法)等,并可于运行时改变fields内容或调用methods。那么我们便可以更灵活的编写代码,代码可以在运行时装配,无需在组件之间进行源代码链接,降低代码的耦合度;还有动态代理的实现等等;但是需要注意的是反射使用不当会造成很高的资源消耗!
三、反射的实现 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 package com.ys.reflex;public class Person { private String name = "Tom" ; public int age = 18 ; public Person () { } private void say () { System.out.println("private say()..." ); } public void work () { System.out.println("public work()..." ); } }
1 2 3 4 5 6 7 8 9 10 11 12 Person p1 = new Person(); Class c1 = p1.getClass(); Class c2 = Person.class; Class c3 = Class.forName("com.ys.reflex.Person" );
3.1 Class类的方法
getName():获得类的完整名字。
getFields():获得类的public类型的属性。
getDeclaredFields():获得类的所有属性。包括private 声明的和继承类
getMethods():获得类的public类型的方法。
getDeclaredMethods():获得类的所有方法。包括private 声明的和继承类
getMethod(String name, Class[] parameterTypes):获得类的特定方法,name参数指定方法的名字,parameterTypes 参数指定方法的参数类型。
getConstructors():获得类的public类型的构造方法。
getConstructor(Class[] parameterTypes):获得类的特定构造方法,parameterTypes 参数指定构造方法的参数类型。
newInstance():通过类的不带参数的构造方法创建这个类的一个对象。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 String className = c2.getName(); System.out.println(className); Field[] fields = c2.getFields(); for (Field field : fields){ System.out.println(field.getName()); } Field [] allFields = c2.getDeclaredFields(); for (Field field : allFields){ System.out.println(field.getName()); } Method [] methods = c2.getMethods(); for (Method method : methods){ System.out.println(method.getName()); } Method [] allMethods = c2.getDeclaredMethods(); for (Method method : allMethods){ System.out.println(method.getName()); } Field f1 = c2.getField("age" ); System.out.println(f1); Field f2 = c2.getDeclaredField("name" ); f2.setAccessible(true ); System.out.println(f2); Object p2 = c2.newInstance(); f2.set(p2,"Bob" ); System.out.println(f2.get(p2)); Constructor [] constructors = c2.getConstructors(); for (Constructor constructor : constructors){ System.out.println(constructor.toString()); }
3.1 获取父类属性 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 public class Parent { public String publicField = "parent_publicField" ; protected String protectField = "parent_protectField" ; String defaultField = "parent_defaultField" ; private String privateField = "parent_privateField" ; } public class Son extends Parent {} public class ReflectionTest { @Test public void testGetParentField () throws Exception { Class c1 = Class.forName("com.ys.model.Son" ); System.out.println(getFieldValue(c1.newInstance(),"privateField" )); } public static Field getDeclaredField (Object obj,String fieldName) { Field field = null ; Class c = obj.getClass(); for (; c != Object.class ; c = c.getSuperclass()){ try { field = c.getDeclaredField(fieldName); field.setAccessible(true ); return field; }catch (Exception e){ } } return null ; } public static Object getFieldValue (Object object,String fieldName) throws Exception { Field field = getDeclaredField(object,fieldName); return field.get(object); } }
通过执行上述代码,我们获得了父类的私有属性值,这里要注意的是直接通过反射获取子类的对象是不能得到父类的属性值的,必须根据反射获得的子类 Class 对象在调用 getSuperclass() 方法获取父类对象,然后在通过父类对象去获取父类的属性值。
灵活使用反射能让我们代码更加灵活,这里比如JDBC原生代码注册驱动,hibernate 的实体类,Spring 的 AOP等等都有反射的实现。但是凡事都有两面性,反射也会消耗系统的性能,增加复杂性等,合理使用才是真!