创建运行时类的对象

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

相关文章

Spring创建Bean的八种主要方式详解

《Spring创建Bean的八种主要方式详解》Spring(尤其是SpringBoot)提供了多种方式来让容器创建和管理Bean,@Component、@Configuration+@Bean、@En... 目录引言一、Spring 创建 Bean 的 8 种主要方式1. @Component 及其衍生注解

使用Java读取本地文件并转换为MultipartFile对象的方法

《使用Java读取本地文件并转换为MultipartFile对象的方法》在许多JavaWeb应用中,我们经常会遇到将本地文件上传至服务器或其他系统的需求,在这种场景下,MultipartFile对象非... 目录1. 基本需求2. 自定义 MultipartFile 类3. 实现代码4. 代码解析5. 自定

MySQL 数据库表操作完全指南:创建、读取、更新与删除实战

《MySQL数据库表操作完全指南:创建、读取、更新与删除实战》本文系统讲解MySQL表的增删查改(CURD)操作,涵盖创建、更新、查询、删除及插入查询结果,也是贯穿各类项目开发全流程的基础数据交互原... 目录mysql系列前言一、Create(创建)并插入数据1.1 单行数据 + 全列插入1.2 多行数据

MySQL 临时表创建与使用详细说明

《MySQL临时表创建与使用详细说明》MySQL临时表是存储在内存或磁盘的临时数据表,会话结束时自动销毁,适合存储中间计算结果或临时数据集,其名称以#开头(如#TempTable),本文给大家介绍M... 目录mysql 临时表详细说明1.定义2.核心特性3.创建与使用4.典型应用场景5.生命周期管理6.注

MySQL的触发器全解析(创建、查看触发器)

《MySQL的触发器全解析(创建、查看触发器)》MySQL触发器是与表关联的存储程序,当INSERT/UPDATE/DELETE事件发生时自动执行,用于维护数据一致性、日志记录和校验,优点包括自动执行... 目录触发器的概念:创建触www.chinasem.cn发器:查看触发器:查看当前数据库的所有触发器的定

创建springBoot模块没有目录结构的解决方案

《创建springBoot模块没有目录结构的解决方案》2023版IntelliJIDEA创建模块时可能出现目录结构识别错误,导致文件显示异常,解决方法为选择模块后点击确认,重新校准项目结构设置,确保源... 目录创建spChina编程ringBoot模块没有目录结构解决方案总结创建springBoot模块没有目录

javaSE类和对象进阶用法举例详解

《javaSE类和对象进阶用法举例详解》JavaSE的面向对象编程是软件开发中的基石,它通过类和对象的概念,实现了代码的模块化、可复用性和灵活性,:本文主要介绍javaSE类和对象进阶用法的相关资... 目录前言一、封装1.访问限定符2.包2.1包的概念2.2导入包2.3自定义包2.4常见的包二、stati

MySQL多实例管理如何在一台主机上运行多个mysql

《MySQL多实例管理如何在一台主机上运行多个mysql》文章详解了在Linux主机上通过二进制方式安装MySQL多实例的步骤,涵盖端口配置、数据目录准备、初始化与启动流程,以及排错方法,适用于构建读... 目录一、什么是mysql多实例二、二进制方式安装MySQL1.获取二进制代码包2.安装基础依赖3.清

在IntelliJ IDEA中高效运行与调试Spring Boot项目的实战步骤

《在IntelliJIDEA中高效运行与调试SpringBoot项目的实战步骤》本章详解SpringBoot项目导入IntelliJIDEA的流程,教授运行与调试技巧,包括断点设置与变量查看,奠定... 目录引言:为良驹配上好鞍一、为何选择IntelliJ IDEA?二、实战:导入并运行你的第一个项目步骤1

IntelliJ IDEA2025创建SpringBoot项目的实现步骤

《IntelliJIDEA2025创建SpringBoot项目的实现步骤》本文主要介绍了IntelliJIDEA2025创建SpringBoot项目的实现步骤,文中通过示例代码介绍的非常详细,对大家... 目录一、创建 Spring Boot 项目1. 新建项目2. 基础配置3. 选择依赖4. 生成项目5.