Java 注解和反射(五)创建运行时类的对象
获取运行时类的完整结构
通过反射获取运行时类的完整结构
Field,Method,Constructor,Superclass,Interface,Annotation
~实现的全部接口
~所继承的父类
~全部的构造器
~全部的方法
~全部的Field
~注解
............
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 | //获得类的信息 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语言访问检查
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 | //动态创建对象,通过反射 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注解和反射(三),根据需要删除无参构造器
分析性能问题
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 | //分析性能问题 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注解和反射(三)
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 阿里最新开源QwQ-32B,效果媲美deepseek-r1满血版,部署成本又又又降低了!
· 开源Multi-agent AI智能体框架aevatar.ai,欢迎大家贡献代码
· Manus重磅发布:全球首款通用AI代理技术深度解析与实战指南
· 被坑几百块钱后,我竟然真的恢复了删除的微信聊天记录!
· 没有Manus邀请码?试试免邀请码的MGX或者开源的OpenManus吧