创建运行时类的对象

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

相关文章

在java中如何将inputStream对象转换为File对象(不生成本地文件)

《在java中如何将inputStream对象转换为File对象(不生成本地文件)》:本文主要介绍在java中如何将inputStream对象转换为File对象(不生成本地文件),具有很好的参考价... 目录需求说明问题解决总结需求说明在后端中通过POI生成Excel文件流,将输出流(outputStre

idea中创建新类时自动添加注释的实现

《idea中创建新类时自动添加注释的实现》在每次使用idea创建一个新类时,过了一段时间发现看不懂这个类是用来干嘛的,为了解决这个问题,我们可以设置在创建一个新类时自动添加注释,帮助我们理解这个类的用... 目录前言:详细操作:步骤一:点击上方的 文件(File),点击&nbmyHIgsp;设置(Setti

Java终止正在运行的线程的三种方法

《Java终止正在运行的线程的三种方法》停止一个线程意味着在任务处理完任务之前停掉正在做的操作,也就是放弃当前的操作,停止一个线程可以用Thread.stop()方法,但最好不要用它,本文给大家介绍了... 目录前言1. 停止不了的线程2. 判断线程是否停止状态3. 能停止的线程–异常法4. 在沉睡中停止5

Spring 中使用反射创建 Bean 实例的几种方式

《Spring中使用反射创建Bean实例的几种方式》文章介绍了在Spring框架中如何使用反射来创建Bean实例,包括使用Class.newInstance()、Constructor.newI... 目录1. 使用 Class.newInstance() (仅限无参构造函数):2. 使用 Construc

C#原型模式之如何通过克隆对象来优化创建过程

《C#原型模式之如何通过克隆对象来优化创建过程》原型模式是一种创建型设计模式,通过克隆现有对象来创建新对象,避免重复的创建成本和复杂的初始化过程,它适用于对象创建过程复杂、需要大量相似对象或避免重复初... 目录什么是原型模式?原型模式的工作原理C#中如何实现原型模式?1. 定义原型接口2. 实现原型接口3

Java实现将byte[]转换为File对象

《Java实现将byte[]转换为File对象》这篇文章将通过一个简单的例子为大家演示Java如何实现byte[]转换为File对象,并将其上传到外部服务器,感兴趣的小伙伴可以跟随小编一起学习一下... 目录前言1. 问题背景2. 环境准备3. 实现步骤3.1 从 URL 获取图片字节数据3.2 将字节数组

在VSCode中本地运行DeepSeek的流程步骤

《在VSCode中本地运行DeepSeek的流程步骤》本文详细介绍了如何在本地VSCode中安装和配置Ollama和CodeGPT,以使用DeepSeek进行AI编码辅助,无需依赖云服务,需要的朋友可... 目录步骤 1:在 VSCode 中安装 Ollama 和 CodeGPT安装Ollama下载Olla

Javascript访问Promise对象返回值的操作方法

《Javascript访问Promise对象返回值的操作方法》这篇文章介绍了如何在JavaScript中使用Promise对象来处理异步操作,通过使用fetch()方法和Promise对象,我们可以从... 目录在Javascript中,什么是Promise1- then() 链式操作2- 在之后的代码中使

MyBatis的配置对象Configuration作用及说明

《MyBatis的配置对象Configuration作用及说明》MyBatis的Configuration对象是MyBatis的核心配置对象,它包含了MyBatis运行时所需的几乎所有配置信息,这个对... 目录MyBATis配置对象Configuration作用Configuration 对象的主要作用C

SpringBoot实现导出复杂对象到Excel文件

《SpringBoot实现导出复杂对象到Excel文件》这篇文章主要为大家详细介绍了如何使用Hutool和EasyExcel两种方式来实现在SpringBoot项目中导出复杂对象到Excel文件,需要... 在Spring Boot项目中导出复杂对象到Excel文件,可以利用Hutool或EasyExcel