创建运行时类的对象

2024-04-28 09:58
文章标签 运行 创建 对象 时类

本文主要是介绍创建运行时类的对象,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

  • 通过反射获取运行时类的完整结构

    Field、Method、Constructor、Superclass、Interface、Annotation

    • 实现的全部接口
    • 所继承的父类
    • 全部的构造器
    • 全部的方法
    • 全部的Field
    • 注解
    public class Test {public static void main(String[] args) throws Exception {//通过反射获取类的Class对象Class c1 = Class.forName("com.java.test.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);}//获得指定属性的值Field name = c1.getDeclaredField("name");System.out.println(name);//获得类的方法Method[] methods = c1.getMethods(); //获得本类及其父类的全部public方法for(Method method : methods){System.out.println("正常的:"+method);}methods = c1.getDeclaredMethods(); //获得本类的所有方法for(Method method : methods){System.out.println("全部的:"+method);}//获得指定方法Method getName = c1.getMethod("getName",null);Method setName = c1.getMethod("setName",String.class);System.out.println(getName);System.out.println(setName);//获得构造器Constructor[] constructors = c1.getConstructors();for (Constructor constructor : constructors) {System.out.println(constructor);}constructors = c1.getDeclaredConstructors();for (Constructor constructor : constructors) {System.out.println("#"+constructor);}//获得指定的构造器Constructor constructor = c1.getDeclaredConstructor(String.class);System.out.println(constructor);}
    }class User{	private String name;public User(String name) {this.name = name;}public User() {}public String getName() {return name;}public void setName(String name) {this.name = name;}@Overridepublic String toString() {return "User [name=" + name + "]";}
    }
    
  • 创建类的对象:调用Class对象的newInstance()方法

    • 类必须有一个无参数的构造器
    • 类的构造器的访问权限需要足够
  • 只有在操作的时候明确的调用类中的构造器,并将参数传递进去之后才可以进行实例化操作

    • 通过Class类的getDeprecatedConstructor(Class… parameterTypes)取得本类的指定参数类型的构造器
    • 向构造器的形参中传递一个对象数组进去,里面包含了构造器中所需的各个参数
    • 通过Constructor实例化对象
    //动态的创建对象,通过反射
    public class Test {public static void main(String[] args) throws Exception {//通过反射获取类的Class对象Class c1 = Class.forName("com.java.test.User");//构造一个对象User user = (User)c1.newInstance(); //本质上是调用类的无参构造器System.out.println(user); //User [name=null]//通过构造器创建对象Constructor constructor = c1.getDeclaredConstructor(String.class);User user2 = (User)constructor.newInstance("小明");System.out.println(user2); //User [name=小明]//通过反射调用普通方法Method setName = c1.getMethod("setName",String.class);//invoke:激活,(对象,"方法的值")setName.invoke(user,"小红");System.out.println(user.getName()); //小红//通过反射操作属性Field name = c1.getDeclaredField("name");//不能直接操作私有属性,需要关闭程序的安全检测name.setAccessible(true); //关闭程序的安全检测name.set(user, "小花");System.out.println(user.getName()); //小花}
    }class User{	//同上
    }
    
  • 分析性能问题

    public class Test{public static void main(String[] args) throws Exception {test01(); //普通方法执行10亿次:3mstest02(); //反射方式执行10亿次:2056mstest03(); //关闭检测执行10亿次:726ms}//普通方式调用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("普通方法执行10亿次:"+(endTime-startTime)+"ms");}//反射方式调用public static void test02() throws Exception {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("反射方式执行10亿次:"+(endTime-startTime)+"ms");}//反射方式调用,关闭检测public static void test03() throws Exception {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("关闭检测执行10亿次:"+(endTime-startTime)+"ms");}
    }
    
  • 反射操作泛型

    • Java采用泛型擦除的机制来引入泛型,Java中的泛型仅仅是给编译器javac使用的,确保数据的安全性和免去强制类型转换问题,但是一旦编译完成,所有和泛型有关的类型全部擦除
    • 为了通过反射操作这些类型,Java新增了ParameterizedType,GenericArrayType,TypeVariable和WildcardType几种类型来代表不能被归一到Class类中的类型但是又和原始类型齐名的类型
      • ParameterizedType:表示一种参数化类型,比如Collection< String >
      • GenericArrayType:表示一种元素类型是参数化类型或者类型变量的数组类型
      • TypeVariable:是各种类型变量的公共父接口
      • WildcardType:代表一种通配符类型表达式
    //通过反射获取泛型
    public class Test{public void test01(Map<String, User>map,List<User> list){System.out.println("test01");}public Map<String, User> test02(){System.out.println("test02");return null;}public static void main(String[] args) throws Exception {Method method = Test.class.getMethod("test01",Map.class,List.class);//获得泛型的参数类型Type[] genericParameterTypes = method.getGenericParameterTypes();for(Type genericParameterType : genericParameterTypes){System.out.println("#" + genericParameterType);if(genericParameterType instanceof ParameterizedType){Type[] actualTypeArguments = ((ParameterizedType)genericParameterType).getActualTypeArguments();for(Type actualTypeArgument :actualTypeArguments){System.out.println(actualTypeArgument);}}}System.out.println("---------");method = Test.class.getMethod("test02", null);//获得返回值泛型参数Type genericReturnType = method.getGenericReturnType();if (genericReturnType instanceof ParameterizedType) {Type[] actualTypeArguments = ((ParameterizedType) genericReturnType).getActualTypeArguments();for (Type actualTypeArgument : actualTypeArguments) {System.out.println(actualTypeArgument);}}}
    }class User{	//同上
    }
    /*
    #java.util.Map<java.lang.String, com.java.test.User>
    class java.lang.String
    class com.java.test.User
    #java.util.List<com.java.test.User>
    class com.java.test.User
    ---------
    class java.lang.String
    class com.java.test.User
    */
    
  • 反射操作注解

    //通过反射操作注解
    public class Test {public static void main(String[] args) throws Exception {Class c1 = Class.forName("com.java.test.Student");//通过反射获得注解Annotation[] annotations = c1.getAnnotations();for (Annotation annotation : annotations) {System.out.println(annotation); //@com.java.test.TableStudent(value=db_student)}//获得注解的value的值TableStudent tableStudent = (TableStudent) c1.getAnnotation(TableStudent.class);String value = tableStudent.value();System.out.println(value); //db_student//获得类指定的注解Field f = c1.getDeclaredField("name");FieldStudent fieldStudent = f.getAnnotation(FieldStudent.class);System.out.println(fieldStudent.columnName()); //db_nameSystem.out.println(fieldStudent.type()); //varcharSystem.out.println(fieldStudent.length()); //3}
    }@TableStudent("db_student")
    class Student {@FieldStudent(columnName = "db_id",type = "int",length = 10)private int id;@FieldStudent(columnName = "db_age",type = "int",length = 10)private int age;@FieldStudent(columnName = "db_name",type = "varchar",length = 3)private String name;public Student() {}public Student(int id, int age, String name) {this.id = id;this.age = age;this.name = name;}public int getId() {return id;}public void setId(int id) {this.id = id;}public int getAge() {return age;}public void setAge(int age) {this.age = age;}public String getName() {return name;}public void setName(String name) {this.name = name;}@Overridepublic String toString() {return "Student [id=" + id + ", age=" + age + ", name=" + name + "]";}}//类名的注解
    @Target(ElementType.TYPE)
    @Retention(RetentionPolicy.RUNTIME)
    @interface TableStudent{String value();
    }//属性的注解
    @Target(ElementType.FIELD)
    @Retention(RetentionPolicy.RUNTIME)
    @interface FieldStudent{String columnName();String type();int length();
    }
    

这篇关于创建运行时类的对象的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/942948

相关文章

Android 悬浮窗开发示例((动态权限请求 | 前台服务和通知 | 悬浮窗创建 )

《Android悬浮窗开发示例((动态权限请求|前台服务和通知|悬浮窗创建)》本文介绍了Android悬浮窗的实现效果,包括动态权限请求、前台服务和通知的使用,悬浮窗权限需要动态申请并引导... 目录一、悬浮窗 动态权限请求1、动态请求权限2、悬浮窗权限说明3、检查动态权限4、申请动态权限5、权限设置完毕后

Python创建Excel的4种方式小结

《Python创建Excel的4种方式小结》这篇文章主要为大家详细介绍了Python中创建Excel的4种常见方式,文中的示例代码简洁易懂,具有一定的参考价值,感兴趣的小伙伴可以学习一下... 目录库的安装代码1——pandas代码2——openpyxl代码3——xlsxwriterwww.cppcns.c

使用Python在Excel中创建和取消数据分组

《使用Python在Excel中创建和取消数据分组》Excel中的分组是一种通过添加层级结构将相邻行或列组织在一起的功能,当分组完成后,用户可以通过折叠或展开数据组来简化数据视图,这篇博客将介绍如何使... 目录引言使用工具python在Excel中创建行和列分组Python在Excel中创建嵌套分组Pyt

通过prometheus监控Tomcat运行状态的操作流程

《通过prometheus监控Tomcat运行状态的操作流程》文章介绍了如何安装和配置Tomcat,并使用Prometheus和TomcatExporter来监控Tomcat的运行状态,文章详细讲解了... 目录Tomcat安装配置以及prometheus监控Tomcat一. 安装并配置tomcat1、安装

mysqld_multi在Linux服务器上运行多个MySQL实例

《mysqld_multi在Linux服务器上运行多个MySQL实例》在Linux系统上使用mysqld_multi来启动和管理多个MySQL实例是一种常见的做法,这种方式允许你在同一台机器上运行多个... 目录1. 安装mysql2. 配置文件示例配置文件3. 创建数据目录4. 启动和管理实例启动所有实例

IDEA运行spring项目时,控制台未出现的解决方案

《IDEA运行spring项目时,控制台未出现的解决方案》文章总结了在使用IDEA运行代码时,控制台未出现的问题和解决方案,问题可能是由于点击图标或重启IDEA后控制台仍未显示,解决方案提供了解决方法... 目录问题分析解决方案总结问题js使用IDEA,点击运行按钮,运行结束,但控制台未出现http://

解决Spring运行时报错:Consider defining a bean of type ‘xxx.xxx.xxx.Xxx‘ in your configuration

《解决Spring运行时报错:Considerdefiningabeanoftype‘xxx.xxx.xxx.Xxx‘inyourconfiguration》该文章主要讲述了在使用S... 目录问题分析解决方案总结问题Description:Parameter 0 of constructor in x

解决IDEA使用springBoot创建项目,lombok标注实体类后编译无报错,但是运行时报错问题

《解决IDEA使用springBoot创建项目,lombok标注实体类后编译无报错,但是运行时报错问题》文章详细描述了在使用lombok的@Data注解标注实体类时遇到编译无误但运行时报错的问题,分析... 目录问题分析问题解决方案步骤一步骤二步骤三总结问题使用lombok注解@Data标注实体类,编译时

JSON字符串转成java的Map对象详细步骤

《JSON字符串转成java的Map对象详细步骤》:本文主要介绍如何将JSON字符串转换为Java对象的步骤,包括定义Element类、使用Jackson库解析JSON和添加依赖,文中通过代码介绍... 目录步骤 1: 定义 Element 类步骤 2: 使用 Jackson 库解析 jsON步骤 3: 添

MySQL分表自动化创建的实现方案

《MySQL分表自动化创建的实现方案》在数据库应用场景中,随着数据量的不断增长,单表存储数据可能会面临性能瓶颈,例如查询、插入、更新等操作的效率会逐渐降低,分表是一种有效的优化策略,它将数据分散存储在... 目录一、项目目的二、实现过程(一)mysql 事件调度器结合存储过程方式1. 开启事件调度器2. 创