深入浅出Java反射:掌握动态编程的艺术
- 小程一言
- 反射
- 何为反射
- 反射核心类
- 反射的基本使用
- 获取`Class`对象
- 创建对象
- 调用方法
- 访问字段
- 示例程序
- 应用场景
- 优缺点分析
- 优点
- 缺点
- 注意
- 再深入一些
- 反射与泛型
- 反射与注解
- 反射与动态代理
- 反射与类加载器
- 结语
小程一言
本专栏是对Java知识点的总结。在学习Java的过程中,学习的笔记,加入自己的思考,结合各种资料的整理。
文章与程序一样,一定都是不完美的,因为不完美,才拥有不断追求完美的动力
以下是符合您要求的博客文章,主类名为crj
,内容全面细致,深度适中,字数约5000字。
反射
Java反射是Java语言中一项强大的功能,它允许程序在运行时动态地获取类的信息并操作类的属性、方法和构造方法。反射机制为Java提供了极大的灵活性,广泛应用于框架开发、动态代理、注解处理等场景。本文将详细介绍Java反射的核心概念、使用方法以及注意事项,并通过示例代码帮助读者更好地理解。
何为反射
简单来说,反射是指在程序运行时,能够动态地获取类的信息(如类名、方法、字段、构造方法等),并能够操作这些信息。通过反射,我们可以在运行时创建对象、调用方法、访问字段,甚至修改私有成员的值。
反射的核心类是java.lang.reflect
包中的Class
、Method
、Field
和Constructor
。通过这些类,我们可以实现动态编程。
反射核心类
Class<T>
: 表示一个类或接口的类型信息。通过Class
对象可以获取类的构造方法、方法和字段。Constructor<T>
: 表示类的构造方法,用于创建对象。Method
: 表示类的方法,用于调用方法。Field
: 表示类的字段,用于访问或修改字段的值。
反射的基本使用
获取Class
对象
要使用反射,首先需要获取目标类的Class
对象。以下是三种常见的获取方式:
Class.forName("全限定类名")
: 通过类的全限定名获取Class
对象。对象.getClass()
: 通过对象实例获取Class
对象。类名.class
: 直接通过类名获取Class
对象。
// 示例:获取String类的Class对象
Class<?> clazz = Class.forName("java.lang.String");
创建对象
通过Class
对象可以获取类的构造方法,并调用newInstance()
方法创建对象。
// 示例:通过反射创建String对象
Class<?> clazz = Class.forName("java.lang.String");
Constructor<?> constructor = clazz.getConstructor(); // 获取无参构造方法
Object obj = constructor.newInstance(); // 创建对象
System.out.println("创建的对象: " + obj);
调用方法
通过Class
对象可以获取类的方法,并调用invoke()
方法执行方法。
// 示例:通过反射调用String的length()方法
Class<?> clazz = Class.forName("java.lang.String");
Method method = clazz.getMethod("length"); // 获取length()方法
int length = (int) method.invoke("Hello"); // 调用方法
System.out.println("字符串长度: " + length);
访问字段
通过Class
对象可以获取类的字段,并访问或修改字段的值。
// 示例:通过反射访问Integer的value字段
Class<?> clazz = Class.forName("java.lang.Integer");
Field field = clazz.getDeclaredField("value"); // 获取value字段
field.setAccessible(true); // 设置可访问私有字段
int value = (int) field.get(10); // 获取字段值
System.out.println("字段值: " + value);
示例程序
以下是一个完整的示例程序,展示了如何使用反射创建对象、调用方法和访问字段。
import java.lang.reflect.Constructor;
import java.lang.reflect.Field;
import java.lang.reflect.Method;public class crj {public static void main(String[] args) {try {// 1. 获取Class对象Class<?> clazz = Class.forName("java.lang.String");// 2. 创建对象Constructor<?> constructor = clazz.getConstructor();Object obj = constructor.newInstance();System.out.println("创建的对象: " + obj);// 3. 调用方法Method method = clazz.getMethod("length");int length = (int) method.invoke("Hello");System.out.println("字符串长度: " + length);// 4. 访问字段Class<?> integerClass = Class.forName("java.lang.Integer");Field field = integerClass.getDeclaredField("value");field.setAccessible(true);int value = (int) field.get(10);System.out.println("Integer的value字段值: " + value);} catch (Exception e) {e.printStackTrace();}}
}
应用场景
- 动态代理: 在运行时创建代理对象,例如Spring AOP。
- 框架开发: 如Spring通过反射管理Bean的生命周期。
- 注解处理: 在运行时读取注解信息,例如JUnit的测试框架。
- 工具开发: 如IDE的代码提示功能。
优缺点分析
优点
- 灵活性高: 可以在运行时动态操作类和方法。
- 功能强大: 适用于框架和工具开发。
缺点
- 性能较低: 反射操作比直接调用慢。
- 破坏封装性: 可以访问私有成员,可能导致安全问题。
- 代码可读性差: 反射代码通常难以理解和维护。
注意
- 性能问题: 反射操作较慢,频繁使用时需谨慎。
- 安全性: 反射可以绕过访问控制,需确保代码的安全性。
- 异常处理: 反射操作可能抛出
IllegalAccessException
、InvocationTargetException
等异常,需妥善处理。
再深入一些
将能联系到的地方都牵连一下,希望能给你更多的思考
反射与泛型
Java反射机制在处理泛型时需要注意类型擦除的问题。由于Java的泛型是通过类型擦除实现的,因此在运行时无法直接获取泛型的具体类型信息。但是,可以通过ParameterizedType
等接口来获取泛型的信息。
import java.lang.reflect.ParameterizedType;
import java.lang.reflect.Type;public class crj{public static class GenericClass<T> {public void printType() {Type type = ((ParameterizedType) getClass().getGenericSuperclass()).getActualTypeArguments()[0];System.out.println("泛型类型: " + type);}}public static void main(String[] args) {GenericClass<String> genericClass = new GenericClass<String>() {};genericClass.printType();}
}
反射与注解
Java反射机制可以用于读取和处理注解。通过反射,我们可以在运行时获取类、方法、字段上的注解信息,并根据注解的值执行相应的逻辑。
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.reflect.Method;@Retention(RetentionPolicy.RUNTIME)
@interface MyAnnotation {String value();
}public class AnnotationReflectionExample {@MyAnnotation("Hello, Annotation!")public void annotatedMethod() {System.out.println("这是一个带有注解的方法");}public static void main(String[] args) throws Exception {Method method = AnnotationReflectionExample.class.getMethod("annotatedMethod");MyAnnotation annotation = method.getAnnotation(MyAnnotation.class);System.out.println("注解值: " + annotation.value());}
}
反射与动态代理
Java反射机制在动态代理中扮演着重要角色。通过Proxy
类和InvocationHandler
接口,我们可以在运行时创建代理对象,并在调用方法时执行额外的逻辑。
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;interface MyInterface {void doSomething();
}public class DynamicProxyExample {public static void main(String[] args) {MyInterface realObject = new MyInterface() {@Overridepublic void doSomething() {System.out.println("真实对象的方法");}};MyInterface proxyObject = (MyInterface) Proxy.newProxyInstance(MyInterface.class.getClassLoader(),new Class<?>[] { MyInterface.class },new InvocationHandler() {@Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {System.out.println("方法调用前");Object result = method.invoke(realObject, args);System.out.println("方法调用后");return result;}});proxyObject.doSomething();}
}
反射与类加载器
Java反射机制与类加载器密切相关。通过自定义类加载器,我们可以在运行时动态加载类,并使用反射机制操作这些类。
import java.io.ByteArrayOutputStream;
import java.io.InputStream;
import java.net.URL;public class CustomClassLoader extends ClassLoader {@Overrideprotected Class<?> findClass(String name) throws ClassNotFoundException {byte[] classData = loadClassData(name);return defineClass(name, classData, 0, classData.length);}private byte[] loadClassData(String className) {InputStream inputStream = getClass().getClassLoader().getResourceAsStream(className.replace('.', '/') + ".class");ByteArrayOutputStream byteArrayOutputStream = new ByteArrayOutputStream();int bufferSize = 4096;byte[] buffer = new byte[bufferSize];int bytesRead;try {while ((bytesRead = inputStream.read(buffer)) != -1) {byteArrayOutputStream.write(buffer, 0, bytesRead);}} catch (Exception e) {e.printStackTrace();}return byteArrayOutputStream.toByteArray();}public static void main(String[] args) throws Exception {CustomClassLoader customClassLoader = new CustomClassLoader();Class<?> clazz = customClassLoader.loadClass("com.example.MyClass");Object obj = clazz.getDeclaredConstructor().newInstance();Method method = clazz.getMethod("myMethod");method.invoke(obj);}
}
结语
Java反射是一项强大的功能,它为Java提供了动态编程的能力。通过反射,我们可以在运行时获取类的信息并操作类的成员。尽管反射具有很高的灵活性,但也存在性能和安全性的问题。在实际开发中,应根据需求合理使用反射,避免滥用。
希望本文能帮助你更好地理解Java反射机制!如果你有任何问题或建议,欢迎在评论区留言。
相关文章:

深入浅出Java反射:掌握动态编程的艺术
小程一言反射何为反射反射核心类反射的基本使用获取Class对象创建对象调用方法访问字段 示例程序应用场景优缺点分析优点缺点 注意 再深入一些反射与泛型反射与注解反射与动态代理反射与类加载器 结语 小程一言 本专栏是对Java知识点的总结。在学习Java的过程中,学习…...

大模型被偷家?CNN结合多模态!
2025深度学习发论文&模型涨点之—— CNN多模态 卷积神经网络是一种特殊类型的神经网络,其主要结构包括卷积层、池化层、全连接层和输出层。卷积层通过卷积操作学习图像的特征,池化层通过下采样操作减少参数数量,全连接层和输出层通过分类…...
UI自动化测试的优缺点?
优点 • 提高测试效率:可以快速地重复执行测试用例。例如,对于一个有大量表单需要验证的网页应用,自动化测试可以在短时间内完成多次输入检查,而手动测试则会花费大量时间。 • 保证测试一致性:每次执行测试的步骤和…...
在 Kubernetes (K8s) 环境中,备份 PostgreSQL 数据库
在 Kubernetes (K8s) 环境中,备份 PostgreSQL 数据库有一些特殊的考虑,因为数据库通常运行在容器中,并且数据存储在卷(如 PersistentVolume)中。你可以通过几种方式在外部备份 PostgreSQL 数据库,下面是一些…...
机器视觉中的3d和2d的区别
在机器视觉中,3D和2D的主要区别体现在数据的维度、处理方式及应用场景上。以下是具体对比: 数据维度 2D视觉 :处理二维图像,仅包含宽度和高度信息,通常以像素矩阵表示。 3D视觉 :处理三维数据,…...

exr 格式下 全景图(经纬图、panorama)转 cubemap
先上效果 (X, -X, Y, -Y, Z, -Z) 下载 exr 经纬图 笔者用的这张:https://polyhaven.com/a/kloofendal_48d_partly_cloudy_puresky 使用 Openexr 的 exrenvmap 工具 下载 我 build 了一份 3.3.2 版本的,免积分下载。 https:/…...

STM32 ADC介绍(硬件原理篇)
目录 背景 AD转换器 采样与保持 量化 编码 AD转换器转换原理 DA转换原理 AD转换原理 背景 在数字系统的广泛应用中,用数字系统处理模拟量的情况十分普遍,因此引入了模拟信号和数字信号的接口问题。为了解决这一问题,首先利用模数转换…...

snort3.0 获取注册规则(19000多条)
面对生活中的手机、电脑网络监控,很多人都是束手无策的,只不过雁过留声风过留痕,黑客路过就会留下入侵痕迹,比如手机没玩的时候,流量异常的增多,并且一直和某一个IP地址通信很频繁,可能是黑客正…...

【GitHub】装修个人主页
持续更新各种好文,长期更新技能手册,建议关注收藏点赞! 创建仓库,仓库名自己用户名 权限:public 勾选add a README file修改README.md 这里都是运用markdown语法,以及html标签编写的,可以自行修…...
名词解释:npm,cnpm,yarn,vite,vue,electron
1. npm (Node Package Manager) 读音: “N-P-M” 或者直接读作 “npm”。 npm 是 Node.js 的官方包管理器,用于安装、发布和管理 JavaScript 软件包。它允许开发者轻松地共享代码,并且可以通过命令行工具来管理依赖关系。通过 npm init 命令可以交互式…...

XMOS的多项音频技术创新将大模型与边缘AI应用密切联系形成生态化合
2025蛇年春节,DeepSeek大语言模型以超低的训练成本震撼全球,预示着大模型技术将以更快的脚步全面走进我们的工作和生活,同时也促进了能够连通各种大模型和应用场景的智能终端将加速演进。语音作为人类与机器最常用的互动沟通媒体,…...

九.Spring Boot使用 ShardingSphere + MyBatis + Druid 进行分库分表
文章目录 前言一、引入依赖二、创建一个light-db_1备用数据库三、配置文件 application-dev.yml四、创建shardingsphere-config.yml完整项目结构 五、测试总结 前言 在现代化微服务架构中,随着数据量的不断增长,单一数据库已难以满足高可用性、扩展性和…...
大数据治理:构建数据驱动的未来基石
一、大数据治理的定义与核心价值 在大数据战略从顶层设计到底层实现的落地过程中,数据治理是基础,技术是承载,分析是手段,应用是目的。与传统数据管理不同,数据治理更侧重于制定行业级制度规范,通过建立数…...
常见的几种设计模式(详细)——应用场景和实现方式
文章目录 🎯单例模式应用实现 🏭工厂模式应用实现 ❓策略模式应用实现 🧑⚖️代理模式应用实现 🔍观察者模式(发布订阅模式)应用实现 🧰装饰器模式应用实现 📰模版方法模式应用实现…...

SonarQube
不同版本的sonarqube需要不同版本的数据库、jdk环境。这个看文档然后确定要求 (有时候文档里标注的系统要求是不行的。比如要求内存2G,但是实际上是不够的,要注意) 我安装的: 官方文档 Prerequisites and overview…...
Nginx 之Rewrite 使用详解
文章目录 1. 概述2. Rewrite 指令 2.1 指令语法2.2 Flag 标记说明 3. Rewrite 与 Location 3.1 Location 分类3.2 Rewrite 和 Location 比较 4. Rewrite 实际场景 4.1 基于域名的跳转4.2 基于客户端 IP 访问跳转4.3 基于参数匹配的跳转4.4 基于目录下所有 PHP 文件跳转4.5 基于…...

注册Gmail如何跳过手机验证环节?
很多小伙伴在注册Gmail的时候都会遇到一个难题:手机号码验证,有可能包括了“手机号无法验证” “国内手机号验证失败” “收不到验证码”等等问题,但 根据真实案例,还有部分人则是“幸运地”没有手机号验证环节,那么今…...
WordPress自助建站全攻略
一、基础概念与核心优势 WordPress自助建站是一种无需编程即可搭建网站的平台,基于开源CMS系统,支持高度定制化。其核心优势主要体现在: 易用性:拖拽式编辑器和直观的后台操作,适合零基础用户快速上手。低成本&#x…...
TreeSet(单列集合)
TreeSet 是 Java 集合框架中的一种实现了 Set 接口的类,它通过一个红黑树(Red-Black Tree)来存储元素。由于使用了树结构,TreeSet 保证了元素的有序性,并且不允许重复元素。 1. TreeSet 的基本特性 有序性࿱…...

Elasticsearch:同义词在 RAG 中重要吗?
作者:来自 Elastic Jeffrey Rengifo 及 Toms Mura 探索 RAG 应用程序中 Elasticsearch 同义词的功能。 同义词允许我们使用具有相同含义的不同词语在文档中搜索,以确保用户无论使用什么确切的词语都能找到他们所寻找的内容。你可能会认为,由于…...

利用最小二乘法找圆心和半径
#include <iostream> #include <vector> #include <cmath> #include <Eigen/Dense> // 需安装Eigen库用于矩阵运算 // 定义点结构 struct Point { double x, y; Point(double x_, double y_) : x(x_), y(y_) {} }; // 最小二乘法求圆心和半径 …...
SkyWalking 10.2.0 SWCK 配置过程
SkyWalking 10.2.0 & SWCK 配置过程 skywalking oap-server & ui 使用Docker安装在K8S集群以外,K8S集群中的微服务使用initContainer按命名空间将skywalking-java-agent注入到业务容器中。 SWCK有整套的解决方案,全安装在K8S群集中。 具体可参…...
MySQL 隔离级别:脏读、幻读及不可重复读的原理与示例
一、MySQL 隔离级别 MySQL 提供了四种隔离级别,用于控制事务之间的并发访问以及数据的可见性,不同隔离级别对脏读、幻读、不可重复读这几种并发数据问题有着不同的处理方式,具体如下: 隔离级别脏读不可重复读幻读性能特点及锁机制读未提交(READ UNCOMMITTED)允许出现允许…...
mongodb源码分析session执行handleRequest命令find过程
mongo/transport/service_state_machine.cpp已经分析startSession创建ASIOSession过程,并且验证connection是否超过限制ASIOSession和connection是循环接受客户端命令,把数据流转换成Message,状态转变流程是:State::Created 》 St…...
ssc377d修改flash分区大小
1、flash的分区默认分配16M、 / # df -h Filesystem Size Used Available Use% Mounted on /dev/root 1.9M 1.9M 0 100% / /dev/mtdblock4 3.0M...
Qt Http Server模块功能及架构
Qt Http Server 是 Qt 6.0 中引入的一个新模块,它提供了一个轻量级的 HTTP 服务器实现,主要用于构建基于 HTTP 的应用程序和服务。 功能介绍: 主要功能 HTTP服务器功能: 支持 HTTP/1.1 协议 简单的请求/响应处理模型 支持 GET…...

从零开始打造 OpenSTLinux 6.6 Yocto 系统(基于STM32CubeMX)(九)
设备树移植 和uboot设备树修改的内容同步到kernel将设备树stm32mp157d-stm32mp157daa1-mx.dts复制到内核源码目录下 源码修改及编译 修改arch/arm/boot/dts/st/Makefile,新增设备树编译 stm32mp157f-ev1-m4-examples.dtb \stm32mp157d-stm32mp157daa1-mx.dtb修改…...

HBuilderX安装(uni-app和小程序开发)
下载HBuilderX 访问官方网站:https://www.dcloud.io/hbuilderx.html 根据您的操作系统选择合适版本: Windows版(推荐下载标准版) Windows系统安装步骤 运行安装程序: 双击下载的.exe安装文件 如果出现安全提示&…...

ios苹果系统,js 滑动屏幕、锚定无效
现象:window.addEventListener监听touch无效,划不动屏幕,但是代码逻辑都有执行到。 scrollIntoView也无效。 原因:这是因为 iOS 的触摸事件处理机制和 touch-action: none 的设置有关。ios有太多得交互动作,从而会影响…...
【Java学习笔记】BigInteger 和 BigDecimal 类
BigInteger 和 BigDecimal 类 二者共有的常见方法 方法功能add加subtract减multiply乘divide除 注意点:传参类型必须是类对象 一、BigInteger 1. 作用:适合保存比较大的整型数 2. 使用说明 创建BigInteger对象 传入字符串 3. 代码示例 import j…...