jvm加载java的过程主要是:

编写java文件-》进行java文件的编译-》生成.class字节码文件-》jvm通过类加载器去加载生成的二进制文件

java编译器将源码文件编译称为二进制的.class文件

为什么要进行字节码增强操作?

不想修改源码,但是又想加入新功能,让程序按照我们的预期去运行,可以通过编译过程和加载过程中去做

相应的操作,

简单来讲就是:将生成的.class文件修改或者替换称为我们需要的目标.class文件。

需要依赖第三方的工具去实现字节码增强

这里使用javassist实现;

步骤:

/**
* 步骤1:编写你最初的业务类,实现业务功能
*/
public class BaseService {
public void basePrint(Map map){
System.out.println(map.toString());
}
}
/**
* 步骤2:编写你需要的对业务类进行增强的类或者内容
*/
public class ExBaseService {
public static void exPrint(Map map){
System.out.println("exbaseService增强的功能==需要增强的方法之前");
} public static void exPrintL(Map map){
System.out.println("exbaseService增强的功能==需要增强的方法之后");
}
}
/**
* 步骤3:编写增强的工具类,之后直接调用后再使用原来的业务类方法就实现增强了
* ClassPool其实是一张保存了CtClass信息的哈希表,key=类的全限定类名,value=类名对应的CtClass对象。
* 当需要对某个类修改的时候,通过方法getCtClass(className)从classpool获取到相应的CtClass
* CtClass:编译时的类信息,一个class文件在代码中的抽象表现形式,全限定类名可以获取CtClass对象,用于表示这个类文件
* CtMethod: 类中的方法 定义或修改
* CtField: 类中的属性 定义或修改
*
* javassist 增强代码片段是字符串编写,以$开头用于表示方法或构造函数参数或方法返回值
*/
public class ServiceUtils {
public static void done() throws NotFoundException, CannotCompileException {
ClassPool classPool = ClassPool.getDefault();//获取默认的类池
//通过全限定类名从类池获取对应需要增强的类
CtClass base = classPool.getOrNull("com.quan.security.intecepter.BaseService");
if (base == null){
System.out.println("can not found");
return;
}
//通过方法getDeclaredMethod和类中需要增强的方法名字得到CtMethod类型的方法抽象
CtMethod basemethod = base.getDeclaredMethod("basePrint");
//组合增强字符串,使用$1获取方法的参数。
StringBuffer sbf = new StringBuffer();
sbf.append("{");
sbf.append("com.quan.security.intecepter.ExBaseService.exPrint($1);");
sbf.append("}"); //进行增强,调用之前增强
basemethod.insertBefore(sbf.toString()); StringBuffer sbf2 = new StringBuffer();
sbf2.append("{");
sbf2.append("com.quan.security.intecepter.ExBaseService.exPrintL($1);");
sbf2.append("}"); //进行增强,调用需要增强方法之后增强
basemethod.insertAfter(sbf2.toString());
//替换增强后的字节码
base.toClass();
}
}
public class IntecepterTest {
/**
* 调用增强工具类,使最初的业务类进行增强,
* 随后调用业务类,查看是否增强
* @param args
*/
public static void main(String[] args) {
try {
ServiceUtils.done();
} catch (NotFoundException e) {
e.printStackTrace();
} catch (CannotCompileException e) {
e.printStackTrace();
}
BaseService baseService = new BaseService();
Map<String,String> map1 = new HashMap<>();
map1.put("name","quan");
baseService.basePrint(map1);
}
}

结果:

exbaseService增强的功能==需要增强的方法之前
{name=quan}
exbaseService增强的功能==需要增强的方法之后

字节码增强-learnning的更多相关文章

  1. 深入浅出Java探针技术1--基于java agent的字节码增强案例

    Java agent又叫做Java 探针,本文将从以下四个问题出发来深入浅出了解下Java agent 一.什么是java agent? Java agent是在JDK1.5引入的,是一种可以动态修改 ...

  2. JVM——字节码增强技术简介

    Java字节码增强指的是在Java字节码生成之后,对其进行修改,增强其功能,这种方式相当于对应用程序的二进制文件进行修改.Java字节码增强主要是为了减少冗余代码,提高性能等. 实现字节码增强的主要步 ...

  3. JVM插庄之一:JVM字节码增强技术介绍及入门示例

    字节码增强技术:AOP技术其实就是字节码增强技术,JVM提供的动态代理追根究底也是字节码增强技术. 目的:在Java字节码生成之后,对其进行修改,增强其功能,这种方式相当于对应用程序的二进制文件进行修 ...

  4. SpringAOP之CGLIB字节码增强

    SpringAOP的基础原理就是动态代理 有两种实现方式:1)jdk动态代理 2)cglib动态代理 jdk动态代理和cglib动态代理的区别在于: cglib没有接口(通过继承父类) 只有实现类.  ...

  5. JDK动态代理和CGLIB字节码增强

    一.JDK动态代理 Java 在 java.lang.reflect 包中有自己的代理支持,该类(Proxy.java)用于动态生成代理类,只需传入目标接口.目标接口的类加载器以及 Invocatio ...

  6. Java字节码增强探秘

    Java字节码增强探秘 https://mp.weixin.qq.com/s/CH9D-E7fxuu462Q2S3t0AA

  7. 从底层入手,解析字节码增强和Btrace应用

    这篇文章聊下字节码和相关的应用. 1.机器码和字节码 机器码(machine code),学名机器语言指令,有时也被称为原生码(Native Code),是电脑的CPU可直接解读的数据. 通常意义上来 ...

  8. 字节码增强技术-Byte Buddy

    本文转载自字节码增强技术-Byte Buddy 为什么需要在运行时生成代码? Java 是一个强类型语言系统,要求变量和对象都有一个确定的类型,不兼容类型赋值都会造成转换异常,通常情况下这种错误都会被 ...

  9. Java字节码增强技术

    简单介绍下几种java字节码增强技术. ASM ASM是一个Java字节码操控框架,它能被用来动态生成类或者增强既有类的功能.ASM可以直接产生class文件,也可以在类被加载入Java虚拟机之前动态 ...

随机推荐

  1. HashMap常用方法(简版)

    HashMap继承自Map,是一个散列表,它的存储形式是键-值对,键值对的类型可以不一样,这是它相较数组方便的一点. 1.put()方法 哈希表.put(key,value): 该方法用于将键值对加入 ...

  2. 【C# 】继承

    背景..什么是继承? 「继承」是对象导向编程的其中一个基本属性. 它可让您定义子类,重复使用(继承).扩充或修改父类别行为. 其成员可供继承的类别称为基底类别. 继承基底类别成员的类别则称为「衍生类别 ...

  3. C#CancellationToken/CancellationTokenSource-取消令牌/取消令牌源 CT/CTS

    详细情况:https://www.cnblogs.com/wucy/p/15128365.html 背景 为什么引入取消令牌? Thread.abort()方法会破坏同步锁中代码的原子逻辑,破坏锁的作 ...

  4. UnboundLocalError: local variable ‘xxx‘ referenced before assignment

    原因 在Python函数中调用了某个和全局变量同名的局部变量,导致编译器不知道此时使用的是全局变量还是局部变量 a = 3 def func(): a+=3 func() UnboundLocalEr ...

  5. Qt:QNetworkRequest

    0.说明 QNetworkRequest类代表被QNetworkAccessManager发送的请求. QNetworkReuqest是网络访问API的一部分,在其内部保留了在网络上发送一个reque ...

  6. 2020.11.2 异步IO 协程

    异步IO 同步IO在一个线程中,CPU执行代码的速度极快,然而,一旦遇到IO操作,如读写文件.发送网络数据时,就需要等待IO操作完成,才能继续进行下一步操作. 在IO操作的过程中,当前线程被挂起,而其 ...

  7. MM32F0020 UART1中断接收

    目录: 1.MM32F0020简介 2.初始化MM32F0020 UART1和NVIC中断 3.编写MM32F0020 UART1中断接收函数 4.编写MM32F0020 UART1发送字节和ASCI ...

  8. PHP动态修改配置文件

    文件结构: index.php 主页 config 配置文件 doUpdate.php 修改功能页 index.php <html> <head> <title>修 ...

  9. PhpStrom 常用的插件

    .env files support 可以在env函数使用是提示.env文件中所有的key值的自动完成功能 Markdown support 在编写.md文件时有预览的功能 PHP composer. ...

  10. 如何实现ARC中weak功能?

    原文链接 我们都知道ARC中weak与assign或者说unsafe_unretained最大的不同就是设置weak属性后,系统会在对象被释放后自动将指向对象的指针置为nil,而assign则会产生一 ...