Java 注解和反射(五)创建运行时类的对象
获取运行时类的完整结构
通过反射获取运行时类的完整结构
Field,Method,Constructor,Superclass,Interface,Annotation
~实现的全部接口
~所继承的父类
~全部的构造器
~全部的方法
~全部的Field
~注解
............
//获得类的信息 public class Test07 { public static void main(String[] args) throws ClassNotFoundException { Class c1 = Class.forName("refection.User"); //获得类的名字 System.out.println(c1.getName());//包名+类名 System.out.println(c1.getSimpleName());//类名 //获得类的属性 Field[] fields = c1.getFields();//只能找到public属性的 fields = c1.getDeclaredFields();//找到全部属性 for (Field field: fields) { System.out.println(field); } //获得类的方法 Method[] methods = c1.getMethods();//获得本类及其父类的public所有方法 methods = c1.getDeclaredMethods();//获得本类的方法 for (Method method: methods) { System.out.println(method); } //获得指定构造器 System.out.println("=============="); Constructor[] constructors = c1.getConstructors();//获得public constructors = c1.getDeclaredConstructors();//获得全部 for (Constructor constructor: constructors) { System.out.println(constructor); } } }
动态创建对象
创建类的对象:调用Class对象的newInstance()方法
1.类必须要有一个无参数构造器
2.类的构造器的访问权限需要足够
思考?==》难道没有无参的构造器就不能创建对象了吗?只要在操作的时候明确的调用类中的构造器,
并将参数传递下去之后,才可以实例化操作
步骤如下:
1.通过Class类的getDeclaredConstructor(Class.....parameterTypes)取得本类的指定形参类型的构造器
2.向构造器的形参中传递一个对象数组进去,里面包含了构造器中所需的各个参数
3.通过Constructor实例化对象
调用指定方法
通过反射,调用类中的方法,通过Method类完成
1.通过Class类的getMethod(String name,Class....parameterTypes)方法取得一个Mehod对象并设置此方法操作时所需要的参数类型
2.之后使用Object invoke(Object obj,Object[] args)进行调用,并向方法中传递要设置的obj对象的参数信息
Object invoke(Object obj,Object[] args)
``Object对应原方法的返回值,若原方法无返回值,此时返回null
``若原方法若为静态方法,此时形参Object obj可谓null
``若原方法形参列表为空,则Object[] args 为null
``若原方法声明为private,则需要在调用此invoke()方法前,显式调用方法对象的setAccessible(true)方法,将能访问private方法
setAccessible
··Method和Field,Constructor对象都有setAccessible()方法
··setAccessible作用是启动和禁用访问安全检查的开关
··参数值为true则指示反射的对象在使用时应该取消Java语言访问检查
1.提高反射效率。如果代码中必须使用反射,而该句代码需要频繁被调用,那么请设置true
2.使得原本无法访问的私有成员也可访问
··参数值为false则指示反射的对象应该实施Java语言访问检查
//动态创建对象,通过反射 public class Test08 { public static void main(String[] args) throws ClassNotFoundException, IllegalAccessException, InstantiationException, NoSuchMethodException, InvocationTargetException, NoSuchFieldException { //获得Class对象 Class c1 = Class.forName("refection.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); //激活(对象,“方法值”) setName.invoke(user3,"小磊"); System.out.println(user3.getName()); //通过反射操作属性 User user4 = (User) c1.newInstance(); Field name = c1.getDeclaredField("name"); //不能直接操作私有属性,需要关闭程序安全检测 name.setAccessible(true);//设置true name.set(user4,"小朱"); System.out.println(user4.getName()); } }
其中的User类见Java注解和反射(三),根据需要删除无参构造器
分析性能问题
//分析性能问题 public class Test09 { //普通方式调用 public static void test01(){ User user = new User(); long startTime = System.currentTimeMillis(); for (int i = 0; i < 1000000000; i++) { user.getName(); } long endTime = System.currentTimeMillis(); System.out.println("普通方式执行十亿次="+(endTime-startTime)+"ms"); } //反射方法调用 public 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 < 1000000000; i++) { getName.invoke(user,null); } long endTime = System.currentTimeMillis(); System.out.println("反射方法执行十亿次="+(endTime-startTime)+"ms"); } //反射方式调用,关闭检测 public 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 < 1000000000; i++) { getName.invoke(user,null); } long endTime = System.currentTimeMillis(); System.out.println("反射方式调用,关闭检测执行十亿次="+(endTime-startTime)+"ms"); } public static void main(String[] args) throws NoSuchMethodException, IllegalAccessException, InvocationTargetException { test01(); test02(); test03(); } }
其中的User类见Java注解和反射(三)