使用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

相关文章

SpringBoot集成XXL-JOB实现任务管理全流程

《SpringBoot集成XXL-JOB实现任务管理全流程》XXL-JOB是一款轻量级分布式任务调度平台,功能丰富、界面简洁、易于扩展,本文介绍如何通过SpringBoot项目,使用RestTempl... 目录一、前言二、项目结构简述三、Maven 依赖四、Controller 代码详解五、Service

Java中HashMap的用法详细介绍

《Java中HashMap的用法详细介绍》JavaHashMap是一种高效的数据结构,用于存储键值对,它是基于哈希表实现的,提供快速的插入、删除和查找操作,:本文主要介绍Java中HashMap... 目录一.HashMap1.基本概念2.底层数据结构:3.HashCode和equals方法为什么重写Has

GO语言中函数命名返回值的使用

《GO语言中函数命名返回值的使用》在Go语言中,函数可以为其返回值指定名称,这被称为命名返回值或命名返回参数,这种特性可以使代码更清晰,特别是在返回多个值时,感兴趣的可以了解一下... 目录基本语法函数命名返回特点代码示例命名特点基本语法func functionName(parameters) (nam

使用shardingsphere实现mysql数据库分片方式

《使用shardingsphere实现mysql数据库分片方式》本文介绍如何使用ShardingSphere-JDBC在SpringBoot中实现MySQL水平分库,涵盖分片策略、路由算法及零侵入配置... 目录一、ShardingSphere 简介1.1 对比1.2 核心概念1.3 Sharding-Sp

Java 正则表达式的使用实战案例

《Java正则表达式的使用实战案例》本文详细介绍了Java正则表达式的使用方法,涵盖语法细节、核心类方法、高级特性及实战案例,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要... 目录一、正则表达式语法详解1. 基础字符匹配2. 字符类([]定义)3. 量词(控制匹配次数)4. 边

Python Counter 函数使用案例

《PythonCounter函数使用案例》Counter是collections模块中的一个类,专门用于对可迭代对象中的元素进行计数,接下来通过本文给大家介绍PythonCounter函数使用案例... 目录一、Counter函数概述二、基本使用案例(一)列表元素计数(二)字符串字符计数(三)元组计数三、C

Java Scanner类解析与实战教程

《JavaScanner类解析与实战教程》JavaScanner类(java.util包)是文本输入解析工具,支持基本类型和字符串读取,基于Readable接口与正则分隔符实现,适用于控制台、文件输... 目录一、核心设计与工作原理1.底层依赖2.解析机制A.核心逻辑基于分隔符(delimiter)和模式匹

Java中的stream流分组示例详解

《Java中的stream流分组示例详解》Java8StreamAPI以函数式风格处理集合数据,支持分组、统计等操作,可按单/多字段分组,使用String、Map.Entry或Java16record... 目录什么是stream流1、根据某个字段分组2、按多个字段分组(组合分组)1、方法一:使用 Stri

Java+AI驱动实现PDF文件数据提取与解析

《Java+AI驱动实现PDF文件数据提取与解析》本文将和大家分享一套基于AI的体检报告智能评估方案,详细介绍从PDF上传、内容提取到AI分析、数据存储的全流程自动化实现方法,感兴趣的可以了解下... 目录一、核心流程:从上传到评估的完整链路二、第一步:解析 PDF,提取体检报告内容1. 引入依赖2. 封装

使用Spring Cache本地缓存示例代码

《使用SpringCache本地缓存示例代码》缓存是提高应用程序性能的重要手段,通过将频繁访问的数据存储在内存中,可以减少数据库访问次数,从而加速数据读取,:本文主要介绍使用SpringCac... 目录一、Spring Cache简介核心特点:二、基础配置1. 添加依赖2. 启用缓存3. 缓存配置方案方案