百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术文章 > 正文

基于 Javassist 实现 Java 动态代理

haoteby 2025-01-03 16:08 1 浏览

阅读完本文你能 get 到的知识点

  • 什么是 Javassist
  • JDK 动态代理
  • 使用 Javassist 实现和 JDK 一样的效果

什么是Javassist

很多同学估计会对这个词有点陌生,但随着你关注的博主越来越多,知道的也越来越多,马上这篇文章就带你走进 Javassist 的世界

Javassist 和 ASM 一样是操作字节码的框架, Javassist 诞生于 1999 年,多少有点年头

使用 Javassist 可以在运行时定义一个新类,可以在 JVM 加载类文件时修改类文件

而且 Javassist 提供不同类型的API: 源码级别字节码级别

本文使用源码级别的 API ,所以你甚至可以在不懂字节码的前提下使用它,入手相对简单

但在性能上略逊于 ASM

JDK 动态代理

话不多说,先来回顾一下我们平时是怎么使用 JDK 动态代理的

Proxy

JDK 提供一个类 Proxy 用于生成代理类

调用类方法 newProxyInstance,传入的参数

  • ClassLoader loader : 定义代理类的类加载器
  • Class<?>[] interfaces : 代理类需要实现的所有接口
  • InvocationHandler h : 调用处理程序,方法调用都会分派到这里

InvocationHandler

InvocationHandler 是一个接口类,定义了调用方法

  • Object proxy : 生成的代理对象
  • Method method : 接口方法实例
  • Object[] args : 方法调用中传递的参数值

如何调用

调用 Proxy.newProxyInstance 生成代理对象, 传入参数接口InvocationHandler实现类的对象处理代理的逻辑

代码设计

在动手写代码之前,我们先花几分钟在脑海中设想一下我们需要生成的代理类是什么样子的?

这里先揭晓了

假设我们定义了一个接口类 LoginService

  • 定义的接口类

那么我们需要生成一个大概是这样的代理类

  • 首先必须得实现了定义的接口 LoginService
  • 接口的所有方法实现都调用都代理到 InvocationHandler 中

几个重要的类

从上面生成的代理类入手,我们生成的类

继承了父类 MObject

实现了需要代理的接口 LoginService

生成了类成员变量 LoginService_0, 这个对应 接口的定义的方法

实现了需要代理的接口方法 login

还有带参数 MInvocationHandler 的构造方法

MObject

MObject 是生成的代理类需要继承的父类,它的作用是存储了 MInvocationHandler(处理程序接口)

MInvocationHandler

同 JDK 自带的接口 InvocationHandler ,用于实现代理方法的处理逻辑

MProxy

同 JDK 自带的类 Proxy

提供生成代理对象的方法 newProxyInstance

编码开始

在经过代码设计之后,我们的脑海里应该有思路了,那就开始动手了

整个过程中比较重要的部分应该就是 MProxy 类了

在 MProxy 里面我们需要实现两大功能:

  • 生成代理类字节码
  • 根据字节码生成对象

生成代理类字节码

  1. 生成代理类名称
  2. 生成空类
  3. 给类设置需要实现的接口
  4. 添加类成员变量
  5. 实现接口方法

生成代理类名称

这一步相对简单,为了防止生成的代理类重名

这里拼接了所有需要代理的接口全限定类名,通过字符串 "_" 连接

生成空类

首先我们需要根据新的类名生成一个空的类,注意类名不要重复了,不然会污染了原有的类

  • ClassPool : 类池,存储所有类的信息,会将类名->类信息 存储到 HashTable 里, 可通过 ClassPool.getDefault() 获取实例
  • CtClass: 代表一个类
  • classPool.makeClass : 生成新的类

给类设置需要实现的接口

这里同样通过 ClassPool 的 get 方法获取到所有传入接口的 CtClass 定义

再调用 setInterfaces 方法给生成的类设置多个接口

添加类成员变量

因为我们调用 MInvocationHandler 的 invoke 方法时需要传入的第二个参数是被代理方法的 Method实例

所以将这个方法的存储到类成员变量中

CtField 代表着一个变量

传入类型、变量名 生成一个 CtField 实例

通过 setModifiers 方法设置变量的修饰符为 static + private

因为这里还要设置变量的值

调用 getFieldInitCode 生成初始化代码

为了获取 Method 对象 ,这里生成了反射的代码去获取

实例: Class.forName(类名).getMethod(方法名, Class<?>... 方法的参数类型);

生成了类成员变量之后,接下来该到实现接口的方法了

实现接口方法

这里需要实现接口的方法

可以通过 CtNewMethod.copy 方法去拷贝需要实现的方法,不要直接使用原来的 CtMethod , 防止污染

拿到新的 CtMethod ,我们需要设置它的方法体、设置修饰符为 public

重点来看看怎么生成方法体代码

这里根据方法返回的类型调用不同的方法

  • void : 没有返回值的 调用 getMethodBodyCodeByVoid
  • 基本数据类型 : 调用 getMethodBodyCodeByPrimitive
  • 其它类型 : 调用 getMethodBodyCode

那按顺序来看,不需要返回值的

那我们需要生成的代码是长这样的

super.h.invoke(this, 对应的类成员变量, new Object[]{方法参数});

这个有个语法需要知道: $0 代表这方法的第一个参数,懂字节码的应该知道非构造方法的第一个入参是 一个隐式的 this ,指向对象本身

new Object[]{} 里面就可以用 11 12 代表着方法的参数了

返回值是基本数据类型的,需要调用调用包装类型对应的拆箱方法 如

Boolean.parseBoolean()

所以和上面生成步骤的区别在于 前后生成了对于基本类型的 parse 代码

最后的返回其它类型的也比较简单

直接生成强转的代码 如 (String)

以上步骤走完,前期准备工作算是做完了,接下来就要根据生成的字节码来实例化对象了

根据字节码生成对象

要根据字节码来生成对象,第一步我们需要编写自定义的类加载器,通过类加载器加载字节码

  1. 编写自定义加载器 MClassLoader ,继承类 ClassLoader
  2. 提供 add 方法将类名映射到对应的字节数组
  3. 重写 ClassLoader 类的 findClass 方法,使用我们生成的字节数组生成类

在 MProxy 中调用 MClassLoader 加载并实例化对象

  1. 加载类 mClassLoader.loadClass(clasName)
  2. 获取带 MInvocationHandler 参数类型的构造
  3. 实例化对象 constructor.newInstance(h)

效果演示

好了,上面的代码已经编写完了,那么现在就来对比一下 JDK 自带的 Proxy 和我们自己实现的 Proxy 的效果

我们定义一个需要代理的接口 LoginService

这里按照 Java 的基本数据类型 以及它们对应的包装类 定义了16个接口方法

分别实现了代理类 CusMInvocationHandler 和 CusJdkInvocationHandler

这两个代理类的实现是一样的

区别在于实现的接口一个是我们定义的 MInvocationHandler

另一个是 JDK 的 InvocationHandler

执行入口类

Main 类分别生成了 Proxy 和 MProxy 的代理对象

然后执行代理对象的各个方法

来看看实现的效果,左边是 JDK 的动态代理,右边是使用 Javassist 实现的动态代理


作者:MinXie
链接:https://juejin.cn/post/7168030376080703495

相关推荐

用户界面干货盘点

为了解决大家找资源难的问题,EVGET特别开辟每周盘点用户界面干货的专栏,一网打尽热门的界面资讯、Demo示例、版本升级及下载、移动Web开发,以及各种UI神器推荐。更多资源及工具也可以在用户界面专题...

不仅仅是创意,26款科技小玩意

新科技不断在卖场出现,总是吸引着消费者的眼球。许多很棒的科技小玩意儿被发明,手机、平板、手提电脑、游戏主机、甚至是3D打印都适用。现在的初创公司已经发正在让21世纪打破各种科技壁垒障碍。本文收集26...

FastReport.Net报表设计器如何连接到SQLCe

MicrosoftSQLServerCompactEdition是一个简单的本地关系数据库,不需要安装,并且已与数据库文件建立连接。您不需要管理员权限即可使用基础功能。您也只能“密码”基础功能...

2015年最值得关注的8款用户界面新品

软件界面开发解决方案这一块一直以来是慧都控件(EVGET)的强项,我们有400多款用户界面产品,250多款图表报表产品,此外还提供专业的软件界面定制开发服务,其中DevExpress定制开发、甘特图定...

小贴士:安装TBarCode office的注意事项和相关资源

TBarCodeoffice是一款适用于MicrosoftWord2007、2010等版本,具有强大功能的条码插件。在这里我们介绍一下安装TBarCodeoffice的注意事项和相关资源。安装...

初学者不容错过的修复Bug小技巧

Bug的发生,我想这是每个开发人员几乎每天都要面对的问题,包括历史上非常有名的编程人员,他们依旧要面对Bug。成为一个熟练的程序员并不意味着永远不会犯错误,而是擅于发现错误并能很好地修正错误。当你刚开...

【推荐】一款基于 .NET 开源的支持多厂区、多项目级的MOM/MES系统

如果您对源码&技术感兴趣,请点赞+收藏+转发+关注,大家的支持是我分享最大的动力!!!项目介绍tmom是一款基于.NET开源、通用的生产制造系统,支持多厂区/多项目级的MOM/MES系统,计划排程...

你不可不知的10个Github功能

Github让全世界的开发人员、设计人员可以在一起工作交流。Github不仅提供大量开源项目、编程语言代码,他也发布过Windows和OSX桌面应用,可以让我们在工作中无缝集成Github。...

Fastreport.Net用户手册(十四):文本编辑

编辑对象的文本,只需双击文本内容,然后会弹出一个文本编辑器。在编辑器右方有一个可以添加至文本中的数据树组件。可以通过鼠标拖拽该组件到需要的地方。在文本中嵌入该组件的另一个方法是双击该组件,然后该组件将...

火狐浏览器开发者专版上手体验

当Mozilla宣布FirefoxDeveloperEdition,我想不少开发者都很高兴,因为第一个大型开发者专用浏览器诞生了。既然是开发者专用版,那么和普通版本肯定是不一样的。早已经迫不及待...

FastReport.Net 2015.3.3 优化了报表解析器

FastReport.Net2015.3.3于近日正式发布。点击FastReport.Net2015.3.3下载试用FastReport.Net最新版本。[Core][Exports]重写保存在...

改变上网体验:10个超赞的Google Chrome扩展

你使用谷歌浏览器浏览网页吗?其实,全世界数以百万的用户都喜欢使用GoogleChrome浏览网页,这也促使其成为全球使用量第二大的Web浏览器。GoogleChrome浏览器具有快速、干净的页面,...

如何在 FastReport Online Designer 中处理报表的 5 个函数

FastReports产品的时代并没有停滞不前。每个月都会添加新的函数和对象,并改进和优化当前的代码。FastReportOnlineDesigner...

Winform应用界面开发技术特点图解

整理一下自己之前的Winform开发要点,以图文的方式展示一些关键性的技术特点,总结一下。...

跨平台的可视化Web报表设计器-FastReport Online Designer

好消息!FastReportOnlineDesigner现在作为一个独立的应用程序发布啦!此前作为FastReport.Net的专业版的一部分的在线设计测试版,现在可以单独或作为FastRepor...