java反射机制基础

简介: java反射机制基础

反射概述


反射就是对于任何一个Class类,在“运行的时候”都是可以直接得到这个类的全部成分的。


在运行时,可以直接得到这个类的构造器对象:Constructor

在运行时,可以直接得到这个类的成员变量对象:Field

在运行时,可以直接得到这个类的成员方法对象:Method

这种运行时动态获取类信息以及动态调用类中成分的能力称为Java语言的反射机制。


反射机制的关键:反射的第一步都是先得到编译后的Class类对象,然后就可以得到Class的全部成分。


HelloWorld.java -> javac -> HelloWorld.class
Class c = HelloWorld.class;


本质上其实就是.class文件就是一个类对象。


反射的基本作用:在运行时获取类的字节码文件对象,然后解析类中的全部成分。

反射的核心思想和关键就是:得到编译以后的class文件对象。


class在内存中的结构


3.png


反射获取类对象


反射的第一步就是要获取Class类的对象

2.png1.png

方式一:Class c1 = Class.forName(“全类名”);

方式二:Class c2 = 类名.class

方式三:Class c3 = 对象.getClass();


@Test
    public void test1() {
        Student student1 = new Student();
        Student student2 = new Student();
//        第一种获取类对象的方法
        Class<?> s1 = null;
        try {
            s1 = Class.forName("Student");
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        }
//        第二种获取类对象的方法
        Class<Student> s2 = Student.class;
//        第三种获取类对象的方法
        Class<? extends Student> s3 = student1.getClass();
        System.out.println(s1);
        System.out.println(s2);
        System.out.println(s3);
    }


12.png


反射获取构造器对象


使用反射技术获取构造器对象并使用


10.png


API


9.png

@Test
    public void test2() throws NoSuchMethodException {
        Class<Student> sc = Student.class;
//        1. 拿到所有的构造器
        Constructor<?>[] constructors = sc.getDeclaredConstructors();
//        输出构造器的名称+参数个数
        for (Constructor<?> constructor : constructors) {
            System.out.println(constructor.getName() + " 参数个数:" + constructor.getParameterCount() + "个");
        }
//        2. 拿到单个构造器
        Constructor<Student> constructor = sc.getDeclaredConstructor(String.class, String.class);
        System.out.println(constructor.getName() + "参数个数:" + constructor.getParameterCount());
    }


使用反射技术获取构造器对象并使用获取到的内容创建出一个对象


反射得到构造器之后的作用仍是创建一个对象,如果说构造器是public,就可以直接new对象,如果说是构造器是私有的private,需要提前将构造器进行暴力反射,再进行构造对象。


反射是可以直接破换掉封装性的,私有的也是可以执行的。

8.png

//        3. 使用无参构造器创建对象
        Student o = (Student)constructors[0].newInstance();
        System.out.println(o);
//        4. 使用全参数构造器创建对象
        constructor.setAccessible(true);  // 暴力反射
        Student student = constructor.newInstance("孙悟空", "20");
        System.out.println(student.getName() + student.getAge());


反射获取成员变量对象


6.png5.png


参数是变量名


4.png


赋值操作时候,需要先选定是哪一个对象,之后将要赋的值给他即可,获取值的时候,需要先找到要获取哪一个对象的值。


//        给成员变量name赋值
        Student student = new Student();  // 首先要new出一个对象来,这样纸才知道是给哪个对象的哪一个成员变量赋值
        field.setAccessible(true);
        field.set(student,"齐天大圣");
        System.out.println(student.getName());
        String name= (String) field.get(student);
        System.out.println(name);


这里也可以使用暴力反射的方式获取。


反射获取方法对象


1.png2.png3.png

@Test
    public  void test5() throws NoSuchMethodException, InvocationTargetException, IllegalAccessException {
        Class<Student> sc = Student.class;
        Method[] methods = sc.getDeclaredMethods();
        for (Method method : methods) {
            String name = method.getName();
            System.out.println(name);
        }
        Method eat = sc.getDeclaredMethod("eat", String.class);
        Student student = new Student();
        Object invoke = eat.invoke(student, "小狗");
        System.out.println(invoke);
        Method eat1 = sc.getDeclaredMethod("eat", String.class,String.class);
        Object invoke1 = eat1.invoke(student, "小狗", "吃好吃的");
        System.out.println(invoke1);
    }

fbcc68c178f5473fac8b8d0fcfaadcb8.png


绕过编译阶段为集合添加数据


反射是作用在运行时的技术,此时集合的泛型将不能产生约束了,此时是可以为集合存入其他任意类型的元素


泛型只是在编译阶段可以约束集合只能操作某种数据类型,在编译成Class文件进入运行阶段的时候,其真实类型都是ArrayList了,泛型相当于被擦除了。


可以通过对象获取class类,之后找到add方法,之后再调用add方法。


@Test
    public void test6() throws NoSuchMethodException, InvocationTargetException, IllegalAccessException {
        ArrayList<String> list = new ArrayList<>();
        list.add("lala");
        System.out.println(list);
        Class aClass = list.getClass();
        Method add = aClass.getDeclaredMethod("add", Object.class);
        add.invoke(list,1);
        System.out.println(list);
    }


反射做通用框架


需求:给你任意一个对象,在不清楚对象字段的情况可以,可以把对象的字段名称和对应值存储到文件中去。


定义一个方法,可以接收任意类的对象。

每次收到一个对象后,需要解析这个对象的全部成员变量名称。

这个对象可能是任意的,那么怎么样才可以知道这个对象的全部成员变量名称呢?

使用反射获取对象的Class类对象,然后获取全部成员变量信息。

遍历成员变量信息,然后提取本成员变量在对象中的具体值

存入成员变量名称和值到文件中去即可。


相关文章
|
1月前
|
监控 Java 应用服务中间件
高级java面试---spring.factories文件的解析源码API机制
【11月更文挑战第20天】Spring Boot是一个用于快速构建基于Spring框架的应用程序的开源框架。它通过自动配置、起步依赖和内嵌服务器等特性,极大地简化了Spring应用的开发和部署过程。本文将深入探讨Spring Boot的背景历史、业务场景、功能点以及底层原理,并通过Java代码手写模拟Spring Boot的启动过程,特别是spring.factories文件的解析源码API机制。
77 2
|
1月前
|
Java 编译器
探索Java中的异常处理机制
【10月更文挑战第35天】在Java的世界中,异常是程序运行过程中不可避免的一部分。本文将通过通俗易懂的语言和生动的比喻,带你了解Java中的异常处理机制,包括异常的类型、如何捕获和处理异常,以及如何在代码中有效地利用异常处理来提升程序的健壮性。让我们一起走进Java的异常世界,学习如何优雅地面对和解决问题吧!
|
22天前
|
Java 程序员
深入理解Java异常处理机制
Java的异常处理是编程中的一块基石,它不仅保障了代码的健壮性,还提升了程序的可读性和可维护性。本文将深入浅出地探讨Java异常处理的核心概念、分类、处理策略以及最佳实践,旨在帮助读者建立正确的异常处理观念,提升编程效率和质量。
108 1
|
23天前
|
Java 开发者 UED
深入探索Java中的异常处理机制##
本文将带你深入了解Java语言中的异常处理机制,包括异常的分类、异常的捕获与处理、自定义异常的创建以及最佳实践。通过具体实例和代码演示,帮助你更好地理解和运用Java中的异常处理,提高程序的健壮性和可维护性。 ##
47 2
|
23天前
|
Java 开发者
Java中的异常处理机制深度剖析####
本文深入探讨了Java语言中异常处理的重要性、核心机制及其在实际编程中的应用策略,旨在帮助开发者更有效地编写健壮的代码。通过实例分析,揭示了try-catch-finally结构的最佳实践,以及如何利用自定义异常提升程序的可读性和维护性。此外,还简要介绍了Java 7引入的多异常捕获特性,为读者提供了一个全面而实用的异常处理指南。 ####
48 2
|
26天前
|
Java 程序员 UED
深入理解Java中的异常处理机制
本文旨在揭示Java异常处理的奥秘,从基础概念到高级应用,逐步引导读者掌握如何优雅地管理程序中的错误。我们将探讨异常类型、捕获流程,以及如何在代码中有效利用try-catch语句。通过实例分析,我们将展示异常处理在提升代码质量方面的关键作用。
35 3
|
26天前
|
Java 数据库连接 开发者
Java中的异常处理机制:深入解析与最佳实践####
本文旨在为Java开发者提供一份关于异常处理机制的全面指南,从基础概念到高级技巧,涵盖try-catch结构、自定义异常、异常链分析以及最佳实践策略。不同于传统的摘要概述,本文将以一个实际项目案例为线索,逐步揭示如何高效地管理运行时错误,提升代码的健壮性和可维护性。通过对比常见误区与优化方案,读者将获得编写更加健壮Java应用程序的实用知识。 --- ####
|
27天前
|
运维 Java 编译器
Java 异常处理:机制、策略与最佳实践
Java异常处理是确保程序稳定运行的关键。本文介绍Java异常处理的机制,包括异常类层次结构、try-catch-finally语句的使用,并探讨常见策略及最佳实践,帮助开发者有效管理错误和异常情况。
86 4
|
26天前
|
开发框架 安全 Java
Java 反射机制:动态编程的强大利器
Java反射机制允许程序在运行时检查类、接口、字段和方法的信息,并能操作对象。它提供了一种动态编程的方式,使得代码更加灵活,能够适应未知的或变化的需求,是开发框架和库的重要工具。
41 2
|
1月前
|
Java
深入探讨Java中的中断机制:INTERRUPTED和ISINTERRUPTED方法详解
在Java多线程编程中,中断机制是协调线程行为的重要手段。了解和正确使用中断机制对于编写高效、可靠的并发程序至关重要。本文将深入探讨Java中的`Thread.interrupted()`和`Thread.isInterrupted()`方法的区别及其应用场景。
37 4