获得类运行时的结构、动态创建对象执行方法和性能对比分析
一、获取运行时类的完整结构
通过反射获取运行时类的完整结构
Field、Method、 Constructor、 Superclass、 Interface、 Annotation
➢实现的全部接口 ➢所继承的父类 ➢全部的构造器
➢全部的方法 ➢全部的Field ➢注解
public class Test08 {public static void main(String[] args) throws ClassNotFoundException, NoSuchFieldException, NoSuchMethodException {Class c1 = Class.forName("com.reflection.User");//获得类的名字System.out.println(c1.getName());//获得包名+类名System.out.println(c1.getSimpleName());//获得类名//获得类的属性System.out.println("==================");Field[] fields = c1.getFields();//只能找到public属性fields = c1.getDeclaredFields();//找到全部属性for (Field field : fields) {System.out.println(field);}//获得指定属性的值Field name = c1.getDeclaredField("name");System.out.println(name);//获得类的方法System.out.println("==============================");Method[] methods = c1.getMethods();//获得本类及父类的全部public方法for (Method method : methods) {System.out.println("正常的"+method);}methods = c1.getDeclaredMethods();//获得本类的所有方法for (Method method : methods) {System.out.println("getDeclaredMethods" + method);}//获得指定方法//重载Method getName = c1.getMethod("getName", null);Method setName = c1.getMethod("setName", String.class);System.out.println(getName);System.out.println(setName);//获得指定的构造器System.out.println("==============================");Constructor[] constructors = c1.getConstructors();for (Constructor constructor : constructors) {System.out.println(constructor);}constructors = c1.getDeclaredConstructors();for (Constructor constructor : constructors) {System.out.println("#" + constructor);}//获得指定的构造器Constructor declaredConstructor = c1.getDeclaredConstructor(String.class, int.class, int.class);System.out.println("指定"+declaredConstructor);}
}二、动态创建对象执行方法
➢创建类的对象:调用Class对象的newInstance()方法
1) 类必须有一个无参数的构造器。
2) 类的构造器的访问权限需要足够
思考? 难道没有无参的构造器就不能创建对象了吗?只要在操作的时候明确的调用类中的构造器,并将参数传递进去之后,才可以实例化操作。
➢步骤如下:
1)通过Class类的getDeclaredConstructor(Class .. parameterTypes)取得本类的指定形参类型的构造器
2)向构造器的形参中传递一个对象数组进去, 里面包含了构造器中所需的各个参数。
3)通过Constructor实例化对象
//动态创建对象,通过反射
public class Test09 {public static void main(String[] args) throws ClassNotFoundException, InstantiationException, IllegalAccessException, NoSuchMethodException, InvocationTargetException, NoSuchFieldException {//获得Class对象Class c1 = Class.forName("com.reflection.User");//构造一个对象//User user =(User) c1.newInstance();//本质上是调用了类的无参构造//System.out.println(user);//通过构造器创建对象//Constructor constructor = c1.getDeclaredConstructor(String.class, int.class, int.class);//User user2 =(User) constructor.newInstance("班班", 001, 18);//System.out.println(user2);//通过反射调用普通方法User user3 = (User) c1.newInstance();//通过反射获取一个方法Method setName = c1.getDeclaredMethod("setName", String.class);//invoke:激活的意思//(对象,“方法的值”)setName.invoke(user3,"班班");System.out.println(user3.getName());//通过反射操作属性System.out.println("===========================");User user4 = (User) c1.newInstance();Field name = c1.getDeclaredField("name");//不能直接操作私有属性,我们需要关闭程序的安全检测,属性或者方法啊的setAccessible(true)。name.setAccessible(true);name.set(user4,"班班");System.out.println(user4.getName());}
}1.调用指定的方法
通过反射,调用类中的方法,通过Method类完成。
①通过Class类的getMethod(String name,Cas...parameter Types)方法取得一个Method对象,并设置此方法操作时所需要的参数类型。
②之后使用Object invoke(Object obj, Object[] args)进行调用,并向方法中传递要设置的obj对象的参数信息。

2.Object invoke(Object obj, Object ... args)
➢Object 对应原方法的返回值,若原方法无返回值,此时返回null
➢若原方法若为静态方法,此时形参Object obj可为null
➢若原方法形参列表为空,则Object[] args为null
➢若原方法声明为private,则需要在调用此invoke()方法前,显式调用方法对象的setAccessible(true)方法,将可访问private的方法。
3.setAccessible
➢Method和Field、Constructor对象都有setAccessible()方法。
➢setAccessible作用是启动和禁用访问安全检查的开关。
➢参数值为true则指示反射的对象在使用时应该取消Java语言访问检查。
—提高反射的效率。如果代码中必须用反射,而该句代码需要频繁的被调用,那么请设置为true。
—使得原本无法访问的私有成员也可以访问
➢参数值为false则指示反射的对象应该实施Java语言访问检查
三、性能对比分析
//分析性能问题
public class Test10 {//普通方式调用 0mspublic static void test01(){User user = new User();long startTime = System.currentTimeMillis();for (int i = 0; i < 1000; i++) {user.getName();}long endtTime = System.currentTimeMillis();System.out.println("普通方式执行一千次:"+(endtTime-startTime) +"ms");}//反射方式调用 2mspublic static void test02() throws NoSuchMethodException, InvocationTargetException, IllegalAccessException {User user = new User();Class c1 = user.getClass();Method getName = c1.getDeclaredMethod("getName", null);long startTime = System.currentTimeMillis();for (int i = 0; i < 1000; i++) {getName.invoke(user,null);}long endtTime = System.currentTimeMillis();System.out.println("反射方式调用执行一千次:"+(endtTime-startTime) +"ms");}//反射方式调用 关闭检测 1mspublic static void test03() throws NoSuchMethodException, InvocationTargetException, IllegalAccessException {User user = new User();Class c1 = user.getClass();Method getName = c1.getDeclaredMethod("getName", null);getName.setAccessible(true);long startTime = System.currentTimeMillis();for (int i = 0; i < 1000; i++) {getName.invoke(user,null);}long endtTime = System.currentTimeMillis();System.out.println("关闭检测执行一千次:"+(endtTime-startTime) +"ms");}public static void main(String[] args) throws InvocationTargetException, NoSuchMethodException, IllegalAccessException {test01();test02();test03();}
}
