前言

最近用了一下午总算把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
相关文章:
上海网站制作网页,上海本地的生活网站有哪些?最好包括生活的各个方面的?
如何用PHP快速搭建CMS系统?
高性能网站服务器部署指南:稳定运行与安全配置优化方案
较简单的网站制作软件有哪些,手机版网页制作用什么软件?
红河网站制作公司,红河事业单位身份证如何上传?
金*站制作公司有哪些,金华教育集团官网?
建站主机功能解析:服务器选择与快速搭建指南
网页制作模板网站推荐,网页设计海报之类的素材哪里好?
建站上传速度慢?如何优化加速网站加载效率?
如何用免费手机建站系统零基础打造专业网站?
小建面朝正北,A点实际方位是否存在偏差?
c++怎么编写动态链接库dll_c++ __declspec(dllexport)导出与调用【方法】
专业网站建设制作报价,网页设计制作要考什么证?
广州网站制作的公司,现在专门做网站的公司有没有哪几家是比较好的,性价比高,模板也多的?
如何在Windows虚拟主机上快速搭建网站?
在线ppt制作网站有哪些,请推荐几个好的课件下载的网站?
建站之星官网登录失败?如何快速解决?
行程制作网站有哪些,第三方机票电子行程单怎么开?
网站制作企业,网站的banner和导航栏是指什么?
零基础网站服务器架设实战:轻量应用与域名解析配置指南
湖南网站制作公司,湖南上善若水科技有限公司做什么的?
网站制作哪家好,cc、.co、.cm哪个域名更适合做网站?
如何选择适配移动端的WAP自助建站平台?
建站之家VIP精选网站模板与SEO优化教程整合指南
如何快速上传建站程序避免常见错误?
建站之星云端配置指南:模板选择与SEO优化一键生成
如何在橙子建站中快速调整背景颜色?
网站制作公司,橙子建站是合法的吗?
香港服务器网站推广:SEO优化与外贸独立站搭建策略
购物网站制作费用多少,开办网上购物网站,需要办理哪些手续?
如何自定义建站之星网站的导航菜单样式?
如何续费美橙建站之星域名及服务?
建站主机无法访问?如何排查域名与服务器问题
建站上市公司网站建设方案与SEO优化服务定制指南
建站之星代理平台如何选择最佳方案?
,购物网站怎么盈利呢?
如何用VPS主机快速搭建个人网站?
ppt制作免费网站有哪些,ppt模板免费下载网站?
如何制作网站标识牌,动态网站如何制作(教程)?
如何快速查询域名建站关键信息?
如何设置并定期更换建站之星安全管理员密码?
齐河建站公司:营销型网站建设与SEO优化双核驱动策略
微信推文制作网站有哪些,怎么做微信推文,急?
如何快速生成凡客建站的专业级图册?
杭州银行网站设计制作流程,杭州银行怎么开通认证方式?
如何在Mac上搭建Golang开发环境_使用Homebrew安装和管理Go版本
深圳网站制作培训,深圳哪些招聘网站比较好?
唐山网站制作公司有哪些,唐山找工作哪个网站最靠谱?
建站之星如何通过成品分离优化网站效率?
攀枝花网站建设,攀枝花营业执照网上怎么年审?
*请认真填写需求信息,我们会在24小时内与您取得联系。