使用javassist动态为已有的类添加方法

2024-05-02 03:04

本文主要是介绍使用javassist动态为已有的类添加方法,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

需求

现在有一个类Person,只有一个int age属性。
现在有一个需求,如下:
动态地为Person类新增increase方法,实现age++的效果。

javassist简介

javassist是一个用于处理java字节码的库,可以动态地修改已有的类的字节码。

javassist与jdk动态代理、cglib的比较

以下是我自己的理解,如有不对,请指正:
javassist动态地修改已有的类的字节码,不是通过生成代码,而是直接修改了类的字节码。这与jdk动态代理、cglib有本质的不同。
因为jdk动态代理是生成原始类的代理,而cglib是对方法进行拦截(gpt给出的方案是对方法进行拦截,我自己也做过实现,如果cglib场景中,Person类不去实现一个有increase方法的接口,使用反射调用的increase方法的时候,直接会报错)。

如果要通过jdk动态代理实现这个需求,则需要Person类实现一个接口,接口有increase方法,可以没有默认实现,否则jdk动态代理不能实现需求。

如果要通过cglib实现需求,则需要Person类实现一个接口,接口有默认方法increase,默认方法可以是空实现,否则cglib不能实现此需求。

javassist应用

前几天看rpc相关的文章,看到dubbo这个rpc是使用的javassist来生成接口的代理,对远程方法进行调用。
为啥dubbo使用javassist来生成带俩,而不是jdk动态代理呢?据说是因为javassis的性能更好,具体为啥更好,后面有空了,可以研究下。

代码

引入maven依赖

<dependencies><dependency><groupId>org.javassist</groupId><artifactId>javassist</artifactId><version>3.28.0-GA</version></dependency><dependency><groupId>org.projectlombok</groupId><artifactId>lombok</artifactId><version>1.18.30</version><scope>compile</scope></dependency>
</dependencies>

创建Person类

import lombok.Data;
import lombok.ToString;@Data
@ToString
public class Person {private int age;
}

测试代码

测试代码中,动态地为Person类新增increase方法,方法中将age+1。

import javassist.*;import java.lang.reflect.Method;public class DynamicMethodGeneratorTest {public static void main(String[] args) throws Exception {// 为Person动态添加方法dynamicAddMethodForPerson();Person person = new Person();// 设置age属性为20person.setAge(20);System.out.println("Before increase: " + person); // 20// 使用反射调用increase方法Method increaseMethod = person.getClass().getMethod("increase");increaseMethod.invoke(person);System.out.println("After increase: " + person); // 21}private static CtClass dynamicAddMethodForPerson() throws NotFoundException, CannotCompileException {ClassPool pool = ClassPool.getDefault();CtClass personClass = pool.get("com.qqcr.train.add_code.Person");// 生成一个新方法increase()CtMethod newMethod = CtNewMethod.make("public void increase() { age++; }", // 方法的作用是为age属性+1personClass);// 将increase方法添加到personClass中。personClass.addMethod(newMethod);/*这行代码必须要调用,调用之后,添加的increase方法才会生效。否则new Person().getClass().getMethod("increase")会报错:NoSuchMethodException。我猜是因为调用了toClass,才会用这里的personClass替换元空间中的Person.class。*/personClass.toClass();return personClass;}
}

测试代码最后的输出

Before increase: Person(age=20)
After increase: Person(age=21)

参考

CHATGTP-POE问答

这篇关于使用javassist动态为已有的类添加方法的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/953261

相关文章

Spring Boot 实现 IP 限流的原理、实践与利弊解析

《SpringBoot实现IP限流的原理、实践与利弊解析》在SpringBoot中实现IP限流是一种简单而有效的方式来保障系统的稳定性和可用性,本文给大家介绍SpringBoot实现IP限... 目录一、引言二、IP 限流原理2.1 令牌桶算法2.2 漏桶算法三、使用场景3.1 防止恶意攻击3.2 控制资源

Mac系统下卸载JAVA和JDK的步骤

《Mac系统下卸载JAVA和JDK的步骤》JDK是Java语言的软件开发工具包,它提供了开发和运行Java应用程序所需的工具、库和资源,:本文主要介绍Mac系统下卸载JAVA和JDK的相关资料,需... 目录1. 卸载系统自带的 Java 版本检查当前 Java 版本通过命令卸载系统 Java2. 卸载自定

springboot下载接口限速功能实现

《springboot下载接口限速功能实现》通过Redis统计并发数动态调整每个用户带宽,核心逻辑为每秒读取并发送限定数据量,防止单用户占用过多资源,确保整体下载均衡且高效,本文给大家介绍spring... 目录 一、整体目标 二、涉及的主要类/方法✅ 三、核心流程图解(简化) 四、关键代码详解1️⃣ 设置

Java Spring ApplicationEvent 代码示例解析

《JavaSpringApplicationEvent代码示例解析》本文解析了Spring事件机制,涵盖核心概念(发布-订阅/观察者模式)、代码实现(事件定义、发布、监听)及高级应用(异步处理、... 目录一、Spring 事件机制核心概念1. 事件驱动架构模型2. 核心组件二、代码示例解析1. 事件定义

SpringMVC高效获取JavaBean对象指南

《SpringMVC高效获取JavaBean对象指南》SpringMVC通过数据绑定自动将请求参数映射到JavaBean,支持表单、URL及JSON数据,需用@ModelAttribute、@Requ... 目录Spring MVC 获取 JavaBean 对象指南核心机制:数据绑定实现步骤1. 定义 Ja

python使用库爬取m3u8文件的示例

《python使用库爬取m3u8文件的示例》本文主要介绍了python使用库爬取m3u8文件的示例,可以使用requests、m3u8、ffmpeg等库,实现获取、解析、下载视频片段并合并等步骤,具有... 目录一、准备工作二、获取m3u8文件内容三、解析m3u8文件四、下载视频片段五、合并视频片段六、错误

Python中提取文件名扩展名的多种方法实现

《Python中提取文件名扩展名的多种方法实现》在Python编程中,经常会遇到需要从文件名中提取扩展名的场景,Python提供了多种方法来实现这一功能,不同方法适用于不同的场景和需求,包括os.pa... 目录技术背景实现步骤方法一:使用os.path.splitext方法二:使用pathlib模块方法三

javax.net.ssl.SSLHandshakeException:异常原因及解决方案

《javax.net.ssl.SSLHandshakeException:异常原因及解决方案》javax.net.ssl.SSLHandshakeException是一个SSL握手异常,通常在建立SS... 目录报错原因在程序中绕过服务器的安全验证注意点最后多说一句报错原因一般出现这种问题是因为目标服务器

Python打印对象所有属性和值的方法小结

《Python打印对象所有属性和值的方法小结》在Python开发过程中,调试代码时经常需要查看对象的当前状态,也就是对象的所有属性和对应的值,然而,Python并没有像PHP的print_r那样直接提... 目录python中打印对象所有属性和值的方法实现步骤1. 使用vars()和pprint()2. 使

CSS实现元素撑满剩余空间的五种方法

《CSS实现元素撑满剩余空间的五种方法》在日常开发中,我们经常需要让某个元素占据容器的剩余空间,本文将介绍5种不同的方法来实现这个需求,并分析各种方法的优缺点,感兴趣的朋友一起看看吧... css实现元素撑满剩余空间的5种方法 在日常开发中,我们经常需要让某个元素占据容器的剩余空间。这是一个常见的布局需求