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 面试高频技术点实操指南详解
本指南结合最新Java技术趋势,涵盖微服务(Spring Cloud Alibaba)、响应式编程(Spring WebFlux)、容器化部署(Docker+Kubernetes)、函数式编程、性能优化及测试等核心领域。通过具体实现步骤与示例代码,深入讲解服务注册发现、配置中心、熔断限流、响应式数据库访问、JVM调优等内容。适合备战Java面试,提升实操能力,助力技术进阶。资源链接:[https://pan.quark.cn/s/14fcf913bae6](https://pan.quark.cn/s/14fcf913bae6)
100 25
|
1月前
|
监控 Java 测试技术
2025 年 Java 核心技术从入门到精通实战指南
《2025年Java核心技术实战指南》全面覆盖Java开发的最新趋势与最佳实践。内容包括Java新特性(如模式匹配、文本块、记录类)、微服务架构(Spring Boot 3.0+、Spring Cloud)、响应式编程(Reactor、WebFlux)、容器化与云原生(Docker、Kubernetes)、数据访问技术(JPA、R2DBC)、函数式编程、单元测试与集成测试(JUnit 5、Mockito)、性能优化与监控等。通过实战案例,帮助开发者掌握构建高性能、高可用系统的技能。代码资源可从[链接](https://pan.quark.cn/s/14fcf913bae6)获取。
126 7
|
1月前
|
IDE Java 开发工具
Java 基础篇必背综合知识点最新技术与实操应用全面总结指南
本总结梳理了Java 17+的核心知识点与新技术,涵盖基础概念(模块化系统、GraalVM)、数据类型(文本块、模式匹配)、流程控制(增强switch)、面向对象(Record类、密封类)、常用类库(Stream API、HttpClient)、实战案例(文件处理)、构建工具(Maven、Gradle)、测试框架(JUnit 5)、开发工具(IDE、Git)及云原生开发(Spring Boot 3、Docker)。通过理论结合实操,帮助开发者掌握Java最新特性并应用于项目中。代码示例丰富,建议配合实践加深理解。
68 4
|
25天前
|
安全 Java API
Java 17 + 特性与现代开发技术实操应用详解
本指南聚焦Java 17+最新技术,涵盖模块化开发、Record类、模式匹配、文本块、Stream API增强、虚拟线程等核心特性,结合Spring Boot 3与Micronaut框架实战。通过实操案例解析现代Java开发技术栈,包括高性能并发编程、GraalVM原生编译及开发工具链配置。同时梳理面试高频考点,助力掌握Java新特性和实际应用,适合学习与项目实践。代码示例丰富,附带完整资源下载链接。
253 0
|
24天前
|
消息中间件 机器学习/深度学习 Java
java 最新技术驱动的智能教育在线实验室设备管理与实验资源优化实操指南
这是一份基于最新技术的智能教育在线实验室设备管理与实验资源优化的实操指南,涵盖系统搭建、核心功能实现及优化策略。采用Flink实时处理、Kafka消息队列、Elasticsearch搜索分析和Redis缓存等技术栈,结合强化学习动态优化资源调度。指南详细描述了开发环境准备、基础组件部署、数据采集与处理、模型训练、API服务集成及性能调优步骤,支持高并发设备接入与低延迟处理,满足教育机构数字化转型需求。代码已提供下载链接,助力快速构建智能化实验室管理系统。
92 44
|
1月前
|
Java API 微服务
2025 年 Java 核心技术全面升级与实战应用详解
这份Java校招实操内容结合了最新技术趋势,涵盖核心技术、微服务架构、响应式编程、DevOps及前沿技术等六大模块。从函数式编程到Spring Cloud微服务,再到容器化与Kubernetes部署,帮助你掌握企业级开发技能。同时,提供AI集成、区块链实践和面试技巧,包括高频算法题与系统设计案例。通过学习这些内容,可应对90%以上的Java校招技术面试,并快速上手实际项目开发。资源链接:[点此获取](https://pan.quark.cn/s/14fcf913bae6)。
223 41
|
25天前
|
SQL Kubernetes Java
Java 最新技术实操:从基础到进阶的详细指南
本文介绍了Java 17及后续版本的核心技术实操,涵盖新特性、集合框架、异常处理和多线程编程等内容。主要包括:密封类(Sealed Classes)的继承层级控制、模式匹配(Pattern Matching)简化类型判断、文本块(Text Blocks)处理多行字符串;集合框架中的工厂方法和Stream API高级操作;异常处理的最佳实践如自动资源管理(ARM)和自定义异常;多线程编程中的CompletableFuture异步编程和ReentrantLock显式锁使用。
85 6
|
27天前
|
安全 Java API
Java最新技术(JDK 11+) 及以上 Java 最新技术之集合框架实操应用详解
本示例基于Java最新技术(JDK 11+),涵盖集合框架的核心功能,结合Java 8+特性(如Stream API、Lambda表达式)与并发编程最佳实践。内容包括:List操作(初始化、Lambda过滤、Stream处理)、Map操作(流式过滤、ConcurrentHashMap原子操作、并行流)、Set操作(TreeSet排序、CopyOnWriteArraySet并发安全)、Queue/Deque操作(优先队列、双端队列)以及高级聚合操作(集合转换、分组统计、平均值计算)。 [代码下载](https://pan.quark.cn/s/14fcf913bae6)
44 4
|
3月前
|
前端开发 Java 物联网
智慧班牌源码,采用Java + Spring Boot后端框架,搭配Vue2前端技术,支持SaaS云部署
智慧班牌系统是一款基于信息化与物联网技术的校园管理工具,集成电子屏显示、人脸识别及数据交互功能,实现班级信息展示、智能考勤与家校互通。系统采用Java + Spring Boot后端框架,搭配Vue2前端技术,支持SaaS云部署与私有化定制。核心功能涵盖信息发布、考勤管理、教务处理及数据分析,助力校园文化建设与教学优化。其综合性和可扩展性有效打破数据孤岛,提升交互体验并降低管理成本,适用于日常教学、考试管理和应急场景,为智慧校园建设提供全面解决方案。
303 70