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

基于 Javassist 实现 Java 动态代理

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

阅读完本文你能 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

相关推荐

能跑源码,还提供数据集:这里有一个入门企业级验证码识别项目

机器之心专栏作者:kerlomz网上关于验证码识别的开源项目众多,但大多是学术型文章或者仅仅是一个测试demo,那么企业级的验证码识别究竟是怎样的呢?1.前言网上关于验证么识别的开源项目众多,但大...

kdj源码_kdj源码公式描述

N:=9;M1:=3;M2:=3;...

QT实现抖动文字和滚动文字,附源码

前言不知道大家有没有发现今天的文章有什么不一样,哈哈,我自己胡拼乱凑弄了一个logo,好不好看就先不说了,最起码萌萌哒...当然这不是今天的重点,在做logo的时候,我原本想让文字动起来的,奈何技术有...

我试图通过这篇文章告诉你,这行源码有多牛逼。

你好呀,我是歪歪。这次给你盘一个特别有意思的源码,正如我标题说的那样:看懂这行源码之后,我不禁鼓起掌来,直呼祖师爷牛逼。...

想了解Python源代码加密吗?现总结如下5大加密混淆手段!

我们在进行...

Android系统基础(03) Android系统源码下载

常规官方网站说明:Android源码官方网站为(google你懂的):https://source.android.com官网参考链接,对应的tag(tag是一种标签,我们可以根据tag来判断下载的...

真香,Python爬取B站弹幕原来如此简单,源码已附在文末

B站的弹幕区一直是人才圣地。今天我就用python来手把手教大家爬取B站排行榜热门视频,Python爬取视频也可以如此简单。...

最详细的 maven 教程,可以收藏_maven步骤

链接|cnblogs.com/hzg110/p/6936101.html正文目前所有的项目都在使用maven,可是一直没有时间去整理学习,这两天正好有时间,好好的整理一下。...

Python黑科技-VIP视频破解源码分享

《利用Python制作自己的VIP视频解析软件》想看的电视剧更新了还要充VIP?喜欢的电影你是VIP还得付费?学了Python哪要这些花里胡哨的,打开我自己的VIP付费视频解析软件,想怎么看就怎么看!...

抖音无水印解析网站源码_抖音无水印解析平台

链接:https://share.weiyun.com/59Ah44S密码:hv4dm7上传到主机解压不用安装,直接打开域名就可以了原文地址:https://www.xigsc.com/post/...

「电脑知识」USBOS 3.0 v2022.1.24 超级PE启动维护工具标准增强版

前几天一直在发PE类工具就是为了制作U盘PE启动重装系统教程的,今天小编继续分享有一篇关于pe的之前小编发布过一款微PE工具箱,今天发布另外一个无任何流氓行为功能超级强大虽然体积大了一点,但是这个...

模版网站建设制作的八步流程_模板的网站

  模版网站比较简单,一般我们按照如下流程就可以制作出来。  一、网站定位:  在建站之前,一定要了解你要建的网站是什么。你必须考虑你网站的标题(关键词)、网站描述以及你想要建立的网站。  二、选择域...

求职季必备,这几个免费的个人简历模板网站,你可千万不要错过!

晃晃悠悠又到了春招的季节,相信一定有很多小伙伴趁着这个金三银四求职季,四处投递简历。这时候一个亮眼优秀的简历,可以很好的祝你吸引HR的注意。今天就把我珍藏很久的5个免费简历模板网站分享给大家,简历模板...

简约时尚作品博客商店网站HTML5模板源码

Meduza是简约时尚和现代的博客HTML模板,带商店电商元素的博客页面。考虑所有的作品集网站需求页可以设计一个旅游网站。原生响应设计HTML5和CSS3(台式机、平板电脑、手机…)简单,干净的和专业...

13 款免费样机网站合集,UI设计、产品设计、VI设计全都有!

俗话说得好,人靠衣装,在作完设计后不少设计师都会为自己的作品套一个「样机」好让设计看过去更加高端大气上档次!今天,我就总结了无论是UI设计、包装设计、服装设计、品牌设计、logo设计,都能用到...