字节码增强-learnning
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的更多相关文章
- 深入浅出Java探针技术1--基于java agent的字节码增强案例
Java agent又叫做Java 探针,本文将从以下四个问题出发来深入浅出了解下Java agent 一.什么是java agent? Java agent是在JDK1.5引入的,是一种可以动态修改 ...
- JVM——字节码增强技术简介
Java字节码增强指的是在Java字节码生成之后,对其进行修改,增强其功能,这种方式相当于对应用程序的二进制文件进行修改.Java字节码增强主要是为了减少冗余代码,提高性能等. 实现字节码增强的主要步 ...
- JVM插庄之一:JVM字节码增强技术介绍及入门示例
字节码增强技术:AOP技术其实就是字节码增强技术,JVM提供的动态代理追根究底也是字节码增强技术. 目的:在Java字节码生成之后,对其进行修改,增强其功能,这种方式相当于对应用程序的二进制文件进行修 ...
- SpringAOP之CGLIB字节码增强
SpringAOP的基础原理就是动态代理 有两种实现方式:1)jdk动态代理 2)cglib动态代理 jdk动态代理和cglib动态代理的区别在于: cglib没有接口(通过继承父类) 只有实现类. ...
- JDK动态代理和CGLIB字节码增强
一.JDK动态代理 Java 在 java.lang.reflect 包中有自己的代理支持,该类(Proxy.java)用于动态生成代理类,只需传入目标接口.目标接口的类加载器以及 Invocatio ...
- Java字节码增强探秘
Java字节码增强探秘 https://mp.weixin.qq.com/s/CH9D-E7fxuu462Q2S3t0AA
- 从底层入手,解析字节码增强和Btrace应用
这篇文章聊下字节码和相关的应用. 1.机器码和字节码 机器码(machine code),学名机器语言指令,有时也被称为原生码(Native Code),是电脑的CPU可直接解读的数据. 通常意义上来 ...
- 字节码增强技术-Byte Buddy
本文转载自字节码增强技术-Byte Buddy 为什么需要在运行时生成代码? Java 是一个强类型语言系统,要求变量和对象都有一个确定的类型,不兼容类型赋值都会造成转换异常,通常情况下这种错误都会被 ...
- Java字节码增强技术
简单介绍下几种java字节码增强技术. ASM ASM是一个Java字节码操控框架,它能被用来动态生成类或者增强既有类的功能.ASM可以直接产生class文件,也可以在类被加载入Java虚拟机之前动态 ...
随机推荐
- Json:Java对象和Json文本转换工具类
Json 是一个用于 Java 对象 和 Json 文本 相互转换的工具类. 安装 下载源码 git clone https://github.com/njdi/durian.git 编译源码 cd ...
- 业务人员可以进行自助ETL操作?这款BI工具你值得拥有
ETL是什么? ETL,是英文 Extract-Transform-Load 的缩写,用来描述将数据从来源端经过抽取(extract).转换(transform).加载(load)至目的端的过程.E ...
- WPF启动屏幕SplashScreen
SplashScreen类为WPF应用程序提供启动屏幕. 方法一:设置图片属性 1. 添加启动图片到项目中 2. 设置图片属性的Build Action为SplashScreen 方法二:编写代码 1 ...
- Python语法3
目录 模块 Python程序架构 第三方工具包: 创建packet包 模块导入方式 异常处理 六种典型异常 异常处理 自定义异常 模块 Python程序架构 Python源代码文件:*.py 一个py ...
- zookeeper的JAVA API使用
1.创建连接 2.创建节点 3.监听信息 Watcher.class 4.获取节点 Stat stat = new Stat(); zk.getData(Path,true,stat); 5.修改节点 ...
- Linux swap分区操作
swap交换分区是系统RAM的补充,swap 分区支持虚拟内存.当没有足够的 RAM 保存系统处理的数据时,会将数据写入 swap 分区,当系统缺乏 swap 空间时,内核会因 RAM 内存耗尽而终止 ...
- BSOJ6310题解
互不相同,太困难啦!!!!!! 考虑可以相同的情况.可以容斥. \[ans=(1+1+1+1)-(2+1+1)+(3+1)+(2+2)-(4) \] 有点抽象,看看就好() \[ans=(a,b,c, ...
- 背包四讲 (AcWing算法基础课笔记整理)
背包四讲 背包问题(Knapsack problem)是一种组合优化的NP完全问题.问题可以描述为:给定一组物品,每种物品都有自己的重量和价格,在限定的总重量内,我们如何选择,才能使得物品的总价格最高 ...
- 面试突击32:为什么创建线程池一定要用ThreadPoolExecutor?
在 Java 语言中,并发编程都是依靠线程池完成的,而线程池的创建方式又有很多,但从大的分类来说,线程池的创建总共分为两大类:手动方式使用 ThreadPoolExecutor 创建线程池和使用 Ex ...
- 配置阿里云RepoForge 镜像
镜像下载.域名解析.时间同步请点击阿里云开源镜像站 一.RepoForge 镜像介绍 Repoforge 是 RHEL 系统下的软件仓库,拥有 10000 多个软件包,被认为是最安全.最稳定的一个软件 ...