java动态修改class_Java Agent入门学习之动态修改代码
前言
最近用了一下午总算把Java agent给跑通了,本篇文章记录一下具体的操作步骤,以免遗忘。下面话不多说,来一起看看详细的介绍:
通过java agent可以动态修改代码(替换、修改类的定义),进行AOP。
目标:
为所有添加@ToString注解的类实现默认的toString方法
需要两个程序,一个是用来测试的程序,一个agent用于修改代码。
1. 测试程序
被测试的程序包括:
- ToString.Java
- Foo.java
- Main.java
具体代码如下:
ToString.java:定义ToString注解
package com.chosen0ne.agent.test;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
@Retention(RetentionPolicy.RUNTIME)
public @interface ToString {
}
Foo.java:很简单用于测试,使用了ToString注解
package com.chosen0ne.agent.test;
@ToString
public class Foo {
}
Main.java:
package com.chosen0ne.agent.test;
public class Main {
public static void main(String[] args) {
Foo foo = new Foo();
System.out.println(foo.toString());
}
}
执行Main.java,结果如下:
com.chosen0ne.agent.test.Foo@7852e922
可以看到toString返回的是Object的默认实现。
2. Agent程序
java agent程序实际上类似于钩子,有两种方式:
- main函数开始前
- 程序运行中
这里主要测试main函数开始前的情况。类似于main函数,需要实现
public static void premain(String agentArgs, Instrumentation inst);
这个函数会在main函数之前被调用。可以在premain中,进行字节码操作,替换或重新实现一些类。这里使用Byte Buddy库,在ASM之上提供了更高级的抽象,便于使用。
具体代码如下:
package com.chosen0ne.ByteCode.agent;
import java.lang.instrument.Instrumentation;
import com.chosen0ne.agent.test.ToString;
import net.bytebuddy.agent.builder.AgentBuilder;
import net.bytebuddy.description.type.TypeDescription;
import net.bytebuddy.dynamic.DynamicType.Builder;
import net.bytebuddy.implementation.FixedValue;
import net.bytebuddy.matcher.ElementMatchers;
public class ToStringAgent {
public static void premain(String args, Instrumentation instrumentation) {
System.out.println("print pre main");
new AgentBuilder.Default()
.type(ElementMatchers.isAnnotatedWith(ToString.class))
.transform(new AgentBuilder.Transformer() {
@Override
public Builder> transform(Builder> builder,
TypeDescription typeDescription, ClassLoader classLoader) {
return builder.method(ElementMatchers.named("toString"))
.intercept(FixedValue.value("test"));
}
}).installOn(instrumentation);
}
}
agent需要打包成jar,并且对于premain的方式需要在MANIFEST.MF中指定Premain-Class,用于指明包含premain函数的类。具体有两种方式打包:
1)直接通过jar命令
编辑生成MANIFEST.MF后,执行:
jar cvfm agent.jar MANIFEST.MF -C . com lib
上述命令打包成的jar包含:
- com:编译生成的class文件
- lib:其依赖的库
2)通过maven直接生成:
通过maven-jar-plugin插件生成jar包,具体配置如下:
org.apache.maven.plugins
maven-jar-plugin
2.1
true
lib/
com.chosen0ne.ByteCode.ByteBuddyTest
com.chosen0ne.ByteCode.agent.ToStringAgent
主要通过manifestEntries标签生成自动的属性,这里指定了Premain-Class
3. 运行
将生成的agent.jar、依赖的ByteBuddy的jar包和测试程序编译生成的class文件放到一个路径下,目录布局如下:
.
├── agent.jar
├── classes
│ └── com
│ └── chosen0ne
│ └── agent
│ └── test
│ ├── Foo.class
│ ├── Main.class
│ └── ToString.class
└── lib
└── byte-buddy-1.2.3.jar
在当前目录执行命令:
java -cp classes:lib/byte-buddy-1.2.3.jar -javaagent:agent.jar com.chosen0ne.agent.test.Main
运行结果如下:
print pre main
test
这里需要注意一点,如果将测试程序也打包成jar包的话,那么在通过-cp指定ByteBuddy库时会失败,找不到对应的class,错误如下:
> java -cp classes:lib/byte-buddy-1.2.3.jar -javaagent:agent.jar -jar agent-test-case-0.0.1-SNAPSHOT.jar
Exception in thread "main" java.lang.NoClassDefFoundError: net/bytebuddy/matcher/ElementMatcher
at java.lang.Class.getDeclaredMethods0(Native Method)
at java.lang.Class.privateGetDeclaredMethods(Class.java:2688)
at java.lang.Class.getDeclaredMethod(Class.java:2115)
at sun.instrument.InstrumentationImpl.loadClassAndStartAgent(InstrumentationImpl.java:327)
at sun.instrument.InstrumentationImpl.loadClassAndCallPremain(InstrumentationImpl.java:401)
Caused by: java.lang.ClassNotFoundException: net.bytebuddy.matcher.ElementMatcher
at java.net.URLClassLoader$1.run(URLClassLoader.java:372)
at java.net.URLClassLoader$1.run(URLClassLoader.java:361)
at java.security.AccessController.doPrivileged(Native Method)
at java.net.URLClassLoader.findClass(URLClassLoader.java:360)
at java.lang.ClassLoader.loadClass(ClassLoader.java:424)
at sun.misc.Launcher$AppClassLoader.loadClass(Launcher.java:308)
at java.lang.ClassLoader.loadClass(ClassLoader.java:357)
... 5 more
FATAL ERROR in native method: processing of -javaagent failed
暂时不知道具体原因。。。所以直接以class运行即可
总结
以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流,谢谢大家对脚本之家的支持。
java动态修改class_Java Agent入门学习之动态修改代码相关推荐
- java -jar 默认参数_JAVA入门学习指南,建议收藏
如果你不懂Java 并且想认真学习接触了解一下Java的语法,建议把这篇文章收藏了,多看几遍,应该可以初步掌握Java 大部分基础的语法 . 让我们出发吧!ps:本文有点长,耐心阅读 . 〇,编程环境 ...
- c和java哪个好学_编程入门学习c++和Java哪个比较好
编程入门学习c++和Java哪个比较好 发布时间:2020-04-25 16:54:41 来源:亿速云 阅读:231 作者:栢白 编程入门学习c++和Java哪个比较好?如今这些都是小白比较关心的,如 ...
- cad修改快捷键_CAD入门学习技巧:CAD软件中的CAD快捷键的分类
很多刚开始CAD入门学习的小伙伴不是很了解CAD快捷键是如何进行分类的?在浩辰CAD软件的庞大的功能下,软件工程师分门别类的实现了各种快捷键的定义和差别.本文将帮助您更进一步成为业内高人. CAD功能 ...
- cad调了比例因子没反应_CAD入门学习之如何修改比例因子
CAD比例因子缩放的方法有很多,在这里我们精简了一部分来给大家一起学习,让更多刚开始进行CAD入门学习的小伙伴能够了解浩辰制图软件中CAD修改比例的方法. CAD修改比例因子的方法如下: 首先使用CA ...
- cad调了比例因子没反应_CAD入门学习之如何修改比例因子?
CAD比例因子缩放的方法有很多,在这里我们精简了一部分来给大家一起学习,让更多刚开始进行CAD入门学习的小伙伴能够了解浩辰制图软件中CAD修改比例的方法. CAD修改比例因子的方法如下: 首先使用浩辰 ...
- 调用java显示有何风险_Java学习----有风险的代码(异常)
Exception继承了Throwable,但是它本身是有异常类的父类. RuntimeException:运行时异常 Exception->RuntimeException->NullP ...
- C、C++、Java到Python,编程入门学习什么语言好?
最近,TIOBE更新了7月的编程语言榜单,常年霸榜的C.Java和Python依然蝉联前三位.万万没想到的是,R语言居然冲到了第八位,创下了史上最佳记录.而且后续随着业内对数据统计和挖掘需求的上涨,R ...
- java actor akka_Java akka框架入门学习
[日期:2020.04.20-2020.04.22] 看完下面的入门介绍之后,再看如下链接(对akka框架整体介绍)https://github.com/guobinhit/akka-guidegi ...
- java的smalltalk规则_Smalltalk 入门学习
前言 为什么要学习 Smalltalk?Smalltalk 是一门历史悠久的语言,虽然如今我们几乎没有机会再去使用它,但它的不少开创性的理念,仍影响了许多现代的编程语言和思想.尤其是如今主流的面向对象 ...
最新文章
- php 派生类 数据库连接 单例模式 xhprof实测 高效连接
- 高速电路中的AC耦合电容
- 你知道“啥是佩奇”,却不一定了解佩奇排名算法
- 【python教程入门学习】机器学习使用Python编程是因为什么?
- 使用Container.ItemIndex获取Repeater、Gridview行的序号的简单方法
- ELK5.3+Kafka集群配置
- 【angularjs】【学习心得】ng-class总结
- 北京联通dns服务器位置,全国联通DNS服务器地址
- EasyExcel一单元格导出多图片
- 几何光学学习笔记(10)- 3.4 理想光学系统的放大率
- 编程打印如下形式的杨辉三角形,打印的杨辉三角形的行数n(不超过10行)要求由用户从键盘输入。要求按照如下函数原型进行编程,不能使用全局变量
- 星起航:抖音小店适合去做吗
- 01博弈三要素和囚徒困境
- c语言中03怎么表示成30,C语言入门篇-03
- 塔望3W消费战略全案|巨头林立,卤味新品牌的破局之路
- 用python做了一个 qq炫舞 机器人
- 共有90款 工作流引擎开源软件
- 树莓派之间如何通讯_树莓派的IO引脚是如何定义的?
- magento1中的eav模式讲解(入表)
- DataGuard搭建