Java中的反射技术

简介: Java的反射机制是在运行状态中,对于任何一个类,都可以知道这个类的所有属性和方法,对于任何一个对象,都可以调用它所有的方法和属性,修改部分类型信息,这种动态获取信息以及动态调用对象方法的功能称为Java的反射机制

🍊一. 反射的概念

Java的反射机制是在运行状态中,对于任何一个类,都可以知道这个类的所有属性和方法,对于任何一个对象,都可以调用它所有的方法和属性,修改部分类型信息,这种动态获取信息以及动态调用对象方法的功能称为Java的反射机制


🍉二. 为什么需要反射?

在日常的第三方开发中,经常遇到某个类的方法或属性是私有的,这时候就可以利用反射机制来获取所需要的私有方法或属性

我们在进行Java程序开发时,为了开发效率,一般会选择IDE开发环境,IDE开发环境有一个强大的功能就是自动提示功能,IDE是如何知道对象中有哪些属性和方法呢?

反射最重要的用途就是开发各种通用框架,比如在spring中,我们将所有的类Bean交给spring容器管理,无论是XML配置Bean还是注解配置,当我们从容器中获取Bean来依赖注入时,容器会读取配置,而配置中给的就是类的信息,spring根据这些信息,需要创建那些Bean,spring就动态的创建这些类


🍋三. 反射的基石

🥕反射的基石是字节码文件对象


Java的源文件是不能直接进行运行的,需要先进行编译为.class的字节码文件,然后使用双亲委派模型被类加载器加载到虚拟机中形成字节码文件对象,才可以在JVM中运行


🍄何时才能触发类的加载呢?只要需要用类就会触发类的加载,比如:


new一个对象的时候

访问一个静态成员的时候

访问一个静态方法的时候

创建一个子类对象的时候

java命令执行一个字节码文件的时候

通过反射机制创建一个字节码文件对象的时候

在Java中,一切皆对象,当字节码文件加载到JVM中,会形成一个Class类对象,即该类在jvm中变成了一个对象


🌽字节码文件对象包含了三部分内容:


构造方法---Constructor对象

成员方法---Method对象

成员变量---Filed对象


🍏四. 反射的实现

反射的第一步就是先获取Class类对象,也就是字节码文件对象,然后通过Class对象的核心方法达到反射的目的


🌾1. 获取字节码文件对象

🏵️获取Class对象有三种方式:


使用Class.forName("类的全路径名"),可能会抛出ClassNotFoundException异常

使用类名.class,需要在编译期间就明确要操作的类

使用对象.getClass()方法,需要先将对象创建出类

先创建一个Student类,将它的属性,方法都设置为私有的

public class Student {
    private String name;
    private int age;
     public Student(String name, int age) {
        this.name = name;
        this.age = age;
    }
    private String getName() {
        return name;
    }
    private void setName(String name) {
        this.name = name;
    }
    private int getAge() {
        return age;
    }
    private void setAge(int age) {
        this.age = age;
    }
    @Override
    public String toString() {
        return "Student{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
}


这时候,Student类的全路径名为:反射枚举lambda.Student

image.png


👁️‍🗨️下面是获取字节码对象三种方式的代码展示:

public class TestReflect {
    public static void main(String[] args) {
        //获取字节码文件对象
        //1.使用Class.forName("类的全路径")
        try {
            Class<?> stuClass1 = Class.forName("反射枚举lambda.Student");
            System.out.println(stuClass1);
            //2.使用类.class
            Class<?> stuClass2 = Student.class;
            System.out.println(stuClass2);
            System.out.println(stuClass1==stuClass2); //true,字节码文件只有一份,故是同一个对象
            //3.使用对象.getClass()
            //该方法需要先创建对象,故先将Student类的构造方法改为公有的再进行下述操作
            Student student = new Student("张三",26);
            Class<?> stuClass3 = student.getClass();
            System.out.println(stuClass3);
            System.out.println(stuClass2==stuClass3); //true,字节码文件只有一份,故是同一个对象
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
        }
    }
}

👁️‍🗨️打印结果:字节码文件只有一份,所以不同方式获得的是同一个对象

image.png


🌴2. 反射的使用

🍁2.1 反射构造方法创建实例

与反射相关的包都在import java.lang.reflect包下面

方法 说明
Constructor[] getConstructors() 获取类中所有公有的构造器对象
Constructor<T> getConstructors(Class...<T> paramTypes) 获取参数匹配的共有的构造器对象
Constructor[] getDeclaredConstructors() 获取类中所有的构造器对象,包括私有的
Constructor<T> getDeclaredConstructors(Class...<T> paramTypes) 获取类中参数匹配的构造器对象,包括私有的


🌻具体步骤:


获取字节码文件对象

使用字节码对象获取构造方法

设置构造方法权限

使用构造方法创建实例对象

👁️‍🗨️代码示例:

public static void main(String[] args) {
        try {
            //1.获取字节码对象
            Class<?> stuClass = Class.forName("反射枚举lambda.Student");
            //2.获取构造方法
            Constructor<?> stuConstructor = stuClass.getDeclaredConstructor(String.class,int.class); //参数也是class类型
            //3.修改方法的访问权限
            stuConstructor.setAccessible(true);
            //4.调用该方法
            Object object = stuConstructor.newInstance("李四",23); //newInstance()创建类的实例,为Object类型 
            Student s = (Student) object;
            System.out.println(s);
        } catch (Exception e) {
            e.printStackTrace();
        }
    }

 

👁️‍🗨️打印结果:

image.png

 

🍀2.2 反射属性

方法 说明
getFields() 获取所有公有的属性对象
getField(String name) 获取某个公有的属性对象
getDeclaredFields() 获取所有的属性对象,包括私有属性
getDeclaredField(String name) 获取某个属性对象,包括私有属性


🌼具体步骤:


获取字节码对象

使用字节码对象获取属性

设置属性权限

调用方法设置属性值

👁️‍🗨️代码示例:

//反射属性
Field sutAge = stuClass.getDeclaredField("age"); //参数为属性
sutAge.setAccessible(true);
sutAge.setInt(s,18); //设置属性值为int,第一个参数为哪个对象,第二个参数为设置值
System.out.println(s);


👁️‍🗨️打印结果:将对象s的age设置为18

image.png


🍂2.3 反射方法

方法 说明
getMethods() 获取该类所有的公有的方法
getMethod(String name,Class...<?> parameterTypes) 获取该类某个公有的方法
getDeclaredMethods() 获取该类所有方法,包括私有
getDeclaredMethod(String name,Class...<?> parameterTypes) 获取该类某个方法,包括私有


🪴具体步骤:


获取字节码对象

使用字节码对象获取方法

设置方法权限

使用方法.invoke调用,第一个参数为哪个对象,后面参数为方法参数的具体值

👁️‍🗨️代码示例:

//反射方法
Method setNameMethod = stuClass.getDeclaredMethod("setName", String.class); //第一个参数为方法名,后面参数为方法参数
setNameMethod.setAccessible(true);
setNameMethod.invoke(s,"王五");
System.out.println(s);

👁️‍🗨️打印结果:将对象s的姓名改为王五

image.png


🥥反射的优缺点

🌹优点:


对于任意一个类,可以获取该类的所有属性和方法,对于一个对象,能调用它任意一个方法

增加程序的灵活性和扩展性,降低耦合性,提高自适应能力

反射已经应用在很多框架中,如:Spring,Struts,Hibernate

🥀缺点:


破坏了类的封装性

使用反射导致程序效率低

反射代码比较复杂,因而会带来维护问题


相关文章
|
1月前
|
存储 监控 安全
单位网络监控软件:Java 技术驱动的高效网络监管体系构建
在数字化办公时代,构建基于Java技术的单位网络监控软件至关重要。该软件能精准监管单位网络活动,保障信息安全,提升工作效率。通过网络流量监测、访问控制及连接状态监控等模块,实现高效网络监管,确保网络稳定、安全、高效运行。
68 11
|
1月前
|
XML Java 编译器
Java注解的底层源码剖析与技术认识
Java注解(Annotation)是Java 5引入的一种新特性,它提供了一种在代码中添加元数据(Metadata)的方式。注解本身并不是代码的一部分,它们不会直接影响代码的执行,但可以在编译、类加载和运行时被读取和处理。注解为开发者提供了一种以非侵入性的方式为代码提供额外信息的手段,这些信息可以用于生成文档、编译时检查、运行时处理等。
73 7
|
1月前
|
移动开发 前端开发 Java
Java最新图形化界面开发技术——JavaFx教程(含UI控件用法介绍、属性绑定、事件监听、FXML)
JavaFX是Java的下一代图形用户界面工具包。JavaFX是一组图形和媒体API,我们可以用它们来创建和部署富客户端应用程序。 JavaFX允许开发人员快速构建丰富的跨平台应用程序,允许开发人员在单个编程接口中组合图形,动画和UI控件。本文详细介绍了JavaFx的常见用法,相信读完本教程你一定有所收获!
Java最新图形化界面开发技术——JavaFx教程(含UI控件用法介绍、属性绑定、事件监听、FXML)
|
17天前
|
监控 JavaScript 数据可视化
建筑施工一体化信息管理平台源码,支持微服务架构,采用Java、Spring Cloud、Vue等技术开发。
智慧工地云平台是专为建筑施工领域打造的一体化信息管理平台,利用大数据、云计算、物联网等技术,实现施工区域各系统数据汇总与可视化管理。平台涵盖人员、设备、物料、环境等关键因素的实时监控与数据分析,提供远程指挥、决策支持等功能,提升工作效率,促进产业信息化发展。系统由PC端、APP移动端及项目、监管、数据屏三大平台组成,支持微服务架构,采用Java、Spring Cloud、Vue等技术开发。
|
29天前
|
Java 数据库连接 Spring
反射-----浅解析(Java)
在java中,我们可以通过反射机制,知道任何一个类的成员变量(成员属性)和成员方法,也可以堆任何一个对象,调用这个对象的任何属性和方法,更进一步我们还可以修改部分信息和。
|
1月前
|
JavaScript 安全 Java
java版药品不良反应智能监测系统源码,采用SpringBoot、Vue、MySQL技术开发
基于B/S架构,采用Java、SpringBoot、Vue、MySQL等技术自主研发的ADR智能监测系统,适用于三甲医院,支持二次开发。该系统能自动监测全院患者药物不良反应,通过移动端和PC端实时反馈,提升用药安全。系统涵盖规则管理、监测报告、系统管理三大模块,确保精准、高效地处理ADR事件。
|
2月前
|
监控 Java
Java基础——反射
本文介绍了Java反射机制的基本概念和使用方法,包括`Class`类的使用、动态加载类、获取方法和成员变量信息、方法反射操作、以及通过反射了解集合泛型的本质。同时,文章还探讨了动态代理的概念及其应用,通过实例展示了如何利用动态代理实现面向切面编程(AOP),例如为方法执行添加性能监控。
|
2月前
|
监控 前端开发 Java
【技术开发】接口管理平台要用什么技术栈?推荐:Java+Vue3+Docker+MySQL
该文档介绍了基于Java后端和Vue3前端构建的管理系统的技术栈及功能模块,涵盖管理后台的访问、登录、首页概览、API接口管理、接口权限设置、接口监控、计费管理、账号管理、应用管理、数据库配置、站点配置及管理员个人设置等内容,并提供了访问地址及操作指南。
|
2月前
|
JSON 前端开发 JavaScript
java-ajax技术详解!!!
本文介绍了Ajax技术及其工作原理,包括其核心XMLHttpRequest对象的属性和方法。Ajax通过异步通信技术,实现在不重新加载整个页面的情况下更新部分网页内容。文章还详细描述了使用原生JavaScript实现Ajax的基本步骤,以及利用jQuery简化Ajax操作的方法。最后,介绍了JSON作为轻量级数据交换格式在Ajax应用中的使用,包括Java中JSON与对象的相互转换。
66 1
|
2月前
|
SQL 监控 Java
技术前沿:Java连接池技术的最新发展与应用
本文探讨了Java连接池技术的最新发展与应用,包括高性能与低延迟、智能化管理和监控、扩展性与兼容性等方面。同时,结合最佳实践,介绍了如何选择合适的连接池库、合理配置参数、使用监控工具及优化数据库操作,为开发者提供了一份详尽的技术指南。
49 7