前言

最近用了一下午总算把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包,具体配置如下:
<build>
<plugins>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-jar-plugin</artifactId>
<version>2.1</version>
<configuration>
<archive>
<manifest>
<addClasspath>true</addClasspath>
<classpathPrefix>lib/</classpathPrefix>
<mainClass>com.chosen0ne.ByteCode.ByteBuddyTest</mainClass>
</manifest>
<manifestEntries>
<Premain-Class>com.chosen0ne.ByteCode.agent.ToStringAgent</Premain-Class>
</manifestEntries>
</archive>
</configuration>
</plugin>
</plugins>
</build>
主要通过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
# 动态修改代码
# agent
# 技术
# agent简介
# Java Agent (代理)探针技术详情
# Java Agents代理是什么
# Java Agent 动态修改字节码详情
# JavaAgent的简单例子
# java agent使用全解析
# java agent 使用及实现代码
# Java Agent探针技术详解示例
# 有两种
# 类似于
# 的是
# 找不到
# 会在
# 用了
# 很简单
# 可以看到
# 这篇文章
# 谢谢大家
# 多说
# 需要注意
# 暂时不
# 操作步骤
# 下午
# 使用了
# 是用来
# 有疑问
# inst
# Transformer
相关文章:
北京制作网站的公司排名,北京三快科技有限公司是做什么?北京三快科技?
网站海报制作教学视频教程,有什么免费的高清可商用图片网站,用于海报设计?
C#怎么创建控制台应用 C# Console App项目创建方法
如何在Windows 2008云服务器安全搭建网站?
详解ASP.NET 生成二维码实例(采用ThoughtWorks.QRCode和QrCode.Net两种方式)
Android使用GridView实现日历的简单功能
php json中文编码为null的解决办法
高防服务器租用首荐平台,企业级优惠套餐快速部署
网站制作外包价格怎么算,招聘网站上写的“外包”是什么意思?
网站制作员失业,怎样查看自己网站的注册者?
Swift中swift中的switch 语句
广州商城建站系统开发成本与周期如何控制?
建站之星如何助力企业快速打造五合一网站?
营销式网站制作方案,销售哪个网站招聘效果最好?
网站制作怎么样才能赚钱,用自己的电脑做服务器架设网站有什么利弊,能赚钱吗?
制作电商网页,电商供应链怎么做?
,想在网上投简历,哪几个网站比较好?
怎么用手机制作网站链接,dw怎么把手机适应页面变成网页?
如何快速搭建高效可靠的建站解决方案?
如何确认建站备案号应放置的具体位置?
代购小票制作网站有哪些,购物小票的简要说明?
nginx修改上传文件大小限制的方法
子杰智能建站系统|零代码开发与AI生成SEO优化指南
小视频制作网站有哪些,有什么看国内小视频的网站,求推荐?
常州自助建站费用包含哪些项目?
如何快速查询网站的真实建站时间?
红河网站制作公司,红河事业单位身份证如何上传?
佛山企业网站制作公司有哪些,沟通100网上服务官网?
如何高效搭建专业期货交易平台网站?
小程序网站制作需要准备什么资料,如何制作小程序?
陕西网站制作公司有哪些,陕西凌云电器有限公司官网?
如何零成本快速生成个人自助网站?
css网站制作参考文献有哪些,易聊怎么注册?
教程网站设计制作软件,怎么创建自己的一个网站?
如何在Windows虚拟主机上快速搭建网站?
想学网站制作怎么学,建立一个网站要花费多少?
,交易猫的商品怎么发布到网站上去?
湖北网站制作公司有哪些,湖北清能集团官网?
婚礼视频制作网站,学习*后期制作的网站有哪些?
MySQL查询结果复制到新表的方法(更新、插入)
青岛网站设计制作公司,查询青岛招聘信息的网站有哪些?
建站之星2.7模板快速切换与批量管理功能操作指南
如何续费美橙建站之星域名及服务?
如何在Windows环境下新建FTP站点并设置权限?
交易网站制作流程,我想开通一个网站,注册一个交易网址,需要那些手续?
如何选购建站域名与空间?自助平台全解析
唐山网站制作公司有哪些,唐山找工作哪个网站最靠谱?
广州网站设计制作一条龙,广州巨网网络科技有限公司是干什么的?
高端建站三要素:定制模板、企业官网与响应式设计优化
建站主机选哪种环境更利于SEO优化?
*请认真填写需求信息,我们会在24小时内与您取得联系。