首页 > 其他分享 >浅谈字节码增强技术系列1-字节码增强概览

浅谈字节码增强技术系列1-字节码增强概览

时间:2022-12-12 10:31:39浏览次数:50  
标签:增强 字节 支持 class String mv public 浅谈

作者:董子龙

前言

前段时间一直想参照lombok的实现原理写一篇可以生成业务单据修改记录插件的专利,再查阅资料的过程中,偶然了解到了字节码增强工具-byteBuddy。但是由于当时时间紧促,所以没有深入的对该组件进行了解。其实再我们的日常开发中,字节码增强组件的身影无处不在,例如spring-aop和mybatis。本着知其然也要知其所以然的精神,我决定沉下心来,对字节码增强技术做一个深入的学习和总结,本文作为该系列的开篇,主要是对字节码做一下简单的介绍,为我们后面的深入学习打下一个好的基础。


一、字节码简述



浅谈字节码增强技术系列1-字节码增强概览_字节码增强





浅谈字节码增强技术系列1-字节码增强概览_java_02



字节码是一种中间状态的二进制文件,是由源码编译过来的,可读性没有源码的高。cpu并不能直接读取字节码,在java中,字节码需要经过JVM转译成机械码之后,cpu才能读取并运行。

使用字节码的好处:一处编译,到处运行。java就是典型的使用字节码作为中间语言,在一个地方编译了源码,拿着.class文件就可以在各种计算机运行。


二、字节码增强的使用场景

如果我们不想修改源码,但是又想加入新功能,让程序按照我们的预期去运行,可以通过编译过程和加载过程中去做相应的操作,简单来讲就是:将生成的.class文件修改或者替换称为我们需要的目标.class文件。

由于字节码增强可以在完全不侵入业务代码的情况下植入代码逻辑,所以可以用它来做一些酷酷的事,比如下面的几种常见场景:

1、动态代理

2、热部署

3、调用链跟踪埋点

4、动态插入log(性能监控)

5、测试代码覆盖率跟踪

...

三、字节码增强的实现方式

字节码工具

类创建

实现接口

方法调用

类扩展

父类方法调用

优点

缺点

常见使用

学习成本

java-proxy

支持

支持

支持

不支持

不支持

简单动态代理首选

功能有限,不支持扩展

spring-aop,MyBatis

1星

asm

支持

支持

支持

支持

支持

任意字节码插入,几乎不受限制

学习难度大,编写代码多

cglib

5星

javaassit

支持

支持

支持

支持

支持

java原始语法,字符串形式插入,写入直观

不支持jdk1.5以上的语法,如泛型,增强for

Fastjson,MyBatis

2星

cglib

支持

支持

支持

支持

支持

与bytebuddy看起来差不多

正在被bytebuddy淘汰

EasyMock,jackson-databind

3星

bytebuddy

支持

支持

支持

支持

支持

支持任意维度的拦截,可以获取原始类、方法,以及代理类和全部参数

不太直观,学习理解有些成本,API非常多

SkyWalking,Mockito,Hibernate,powermock

3星

四、简单示例

AOP是我们在日常开发中常用的架构设计思想,AOP的主要的实现有cglib,Aspectj,Javassist,java proxy等。接下来,我们就以我们日常开发中会遇到的在方法执行前后打印日志为切入点,手动用字节码来实现一下AOP。

定义目标接口与实现

public class SayService{
public void say(String str) {
System.out.println("hello" + str);
}
}

定义了类SayService,再执行say方法之前,我们会打印方法开始执行start,方法执行之后,我们会打印方法执行结束end

ASM实现AOP

4.1.1、引入jar包
<dependency>    
<groupId>org.ow2.asm</groupId>
<artifactId>asm</artifactId>
<version>9.1</version>
</dependency>
4.1.2、AOP具体实现
public class ResourceClassVisitor extends ClassVisitor implements Opcodes {

public ResourceClassVisitor(ClassVisitor cv) {
super(Opcodes.ASM4, cv);
}

public ResourceClassVisitor(int i, ClassVisitor classVisitor) {
super(i, classVisitor);
}

/**访问类基本信息*/
@Override
public void visit(int version, int access, String name,
String signature, String superName, String[] interfaces) {
this.cv.visit(version, access, name, signature, superName, interfaces);
}

/**访问方法基本信息*/
@Override
public MethodVisitor visitMethod(int access, String name,
String desc, String signature, String[] exceptions) {
MethodVisitor mv = this.cv.visitMethod(access, name, desc,
signature, exceptions);
//假如不是构造方法,我们构建方法的访问对象(MethodVisitor)
if (!name.equals("<init>") && mv != null) {
mv = new ResourceClassVisitor.MyMethodVisitor((MethodVisitor)mv);
}

return (MethodVisitor)mv;
}

/**自定义方法访问对象*/
class MyMethodVisitor extends MethodVisitor implements Opcodes {

public MyMethodVisitor(MethodVisitor mv) {
super(Opcodes.ASM4, mv);
}
/**此方法会在方法执行之前执行*/
@Override
public void visitCode() {
super.visitCode();
this.mv.visitFieldInsn(Opcodes.GETSTATIC, "java/lang/System", "out",
"Ljava/io/PrintStream;");
this.mv.visitLdcInsn("方法开始执行start");
this.mv.visitMethodInsn(Opcodes.INVOKEVIRTUAL, "java/io/PrintStream",
"println", "(Ljava/lang/String;)V", false);
}
/**对应方法体本身*/
@Override
public void visitInsn(int opcode) {
//在方法return或异常之前,添加一个end输出
if ((opcode >= Opcodes.IRETURN && opcode <= Opcodes.RETURN) || opcode == Opcodes.ATHROW) {
this.mv.visitFieldInsn(Opcodes.GETSTATIC, "java/lang/System", "out",
"Ljava/io/PrintStream;");
this.mv.visitLdcInsn("方法执行结束end");
this.mv.visitMethodInsn(Opcodes.INVOKEVIRTUAL, "java/io/PrintStream",
"println", "(Ljava/lang/String;)V", false);
}
this.mv.visitInsn(opcode);
}
}
}
public class AopTest {

public static void main(String[] args) throws IOException {
//第一步:构建ClassReader对象,读取指定位置的class文件(默认是类路径-classpath)
ClassReader classReader = new ClassReader("com/aop/SayService");
//第二步:构建ClassWriter对象,基于此对象创建新的class文件
//ClassWriter.COMPUTE_FRAMES 表示ASM会自动计算max stacks、max locals和stack map frame的具体内容。
//ClassWriter.COMPUTE_MAXS 表示ASM会自动计算max stacks和max locals,但不会自动计算stack map frames。
ClassWriter classWriter = new ClassWriter(ClassWriter.COMPUTE_FRAMES);//推荐使用COMPUTE_FRAMES
//第三步:构建ClassVisitor对象,此对象用于接收ClassReader对象的数据,并将数据处理后传给ClassWriter对象
ClassVisitor classVisitor = new ResourceClassVisitor(classWriter);
//第四步:基于ClassReader读取class信息,并将数据传递给ClassVisitor对象
//这里的参数ClassReader.SKIP_DEBUG表示跳过一些调试信息等,ASM代码看上去就会更简洁
//这里的参数ClassReader.SKIP_FRAMES表示跳过一些方法中的部分栈帧信息,栈帧手动计算非常复杂,所以交给系统去做吧
//推荐用这两个参数
classReader.accept(classVisitor, ClassReader.SKIP_DEBUG|ClassReader.SKIP_FRAMES);
//第五步:从ClassWriter拿到数据,并将数据写出到一个class文件中
byte[] data = classWriter.toByteArray();
//将字节码写入到磁盘的class文件
File f = new File("target/classes/com/aop/SayService.class");
FileOutputStream fout = new FileOutputStream(f);
fout.write(data);
fout.close();
SayService rs = new SayService();
rs.say("asm");//start,handle(),end
}
}
4.1.3、测试类输出结果



浅谈字节码增强技术系列1-字节码增强概览_字节码增强_03



Javassist实现AOP


4.2.1、引入jar包
<dependency>
<groupId>org.javassist</groupId>
<artifactId>javassist</artifactId>
<version>3.28.0-GA</version>
</dependency>
4.2.2、AOP具体实现
public class AopTest {

public static void main(String[] args) throws Exception {
ClassPool pool = ClassPool.getDefault();
CtClass cc = pool.get("com.aop.SayService");
CtMethod personFly = cc.getDeclaredMethod("say");
personFly.insertBefore("System.out.println(\"方法开始执行start\");");
personFly.insertAfter("System.out.println(\"方法执行结束end\");");
cc.toClass();
SayService sayService = new SayService();
sayService.say("assist");
}
}
4.2.3、测试类输出结果



浅谈字节码增强技术系列1-字节码增强概览_java_04



五、总结

作为字节码增强系列文章的开篇,只是简单的介绍了一下字节码的定义、字节码的实现方式,最后通过具体示例向大家展示了如何对字节码进行增强。再后续的文章中,会对相关框架的原理及具体应用做一个细化的总结,欢迎各位大佬的批评与指正。

标签:增强,字节,支持,class,String,mv,public,浅谈
From: https://blog.51cto.com/u_15714439/5929153

相关文章

  • 浅谈古建筑电气火灾的防控与管理
    罗轩志安科瑞电气股份有限公司 上海嘉定201801摘要:我国古建筑多为砖木结构,当发生火灾事故时具有蔓延快、扑救难的特点,而火灾对古建筑的损害性很大,电气火灾事故在我国火灾......
  • 《激励》读后浅谈--结合自己的一些实际管理经验讲点能懂的话
        人力资源是一个企业最重要的资源之一,其也可以作为公司核心竞争的重要因素。    公司要发展取决于人,所以公司员工的素质和活力是一个企业的根本动力,也是一个企业......
  • Java 字节码增强技术小记
    总结是什么:字节码增强技术用于对静态字节码文件进行修改或动态修改运行中的程序。使用场景有【动态代理、AOP、ORM框架】、【热部署、Mock测试、性能诊断工具】等,例如动......
  • 2021 届 字节跳动 校招提前批开始啦~~~
    又是一年校招季~~字节跳动的2021届校招提前批已经开始了~~因为疫情,今年各家公司的大规模校园宣讲多多少少都会受到影响。所以大家一定要特别关注校招的线上宣传活动,公众号......
  • 浅谈Redis大Key与热Key
    如何定义大Key和热Key如何定义大Key如何定义热Key大Key和热Key产生的原因大Key和热Key有哪些危害大Key的危害热Key的危害如何发现大Ke......
  • 浅谈“分层图”思想
    “分层图”思想前置芝士:最短路算法所谓分层图,即把一整个完整的图,分为若干层,每层对应着原图中的某种状态。分层图每一层大致有如下特点:每一层极为相似甚至相同,以至于......
  • io流之【字节/字符缓冲流】
    缓冲流,也叫高效流,是对4个基本的Filexxx流的增强,所以也是4个流,按照数据类型分类:字节缓冲流:BufferedInputStream,BufferedOutputStream字符缓冲流:BufferedReader,Buffered......
  • io流之字节流【基础流】
    ①字节输入流:读取文件中的数据packagecom.Lucky.io.byteStream;importjava.io.FileInputStream;importjava.io.FileNotFoundException;importjava.io.IOExceptio......
  • easylogging++的那些事(四)源码分析(十五)浅谈easylogging++的设计理念
    目录面向过程面向对象Writer类PErrorWriter类Helpers和Loggers类泛型编程IterableContainer抽象类模板,IterablePriorityQueue子类模板,IterableQueue子类模板和......
  • java基础,注释,标识符。数据类型,字节
    一.注释注释不会被执行,是我们写代码的人看的书写注释是一个非常好的习惯平时写代码一定要注意规范java的注释有三种单行注释多行注释文档注释public......