用了一下午總算把java agent給跑通了,本篇文章記錄一下具體的操作步驟,以免遺忘。。。
通過java agent可以動態修改代碼(替換、修改類的定義),進行AOP。
目標:
為所有添加@ToString注解的類實現默認的toString方法
被測試的程序包括:
- 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 {
}
package com.chosen0ne.agent.test;
@ToString
public class Foo {
}
package com.chosen0ne.agent.test;
public class Main {
public static void main(String[] args) {
Foo foo = new Foo();
System.out.println(foo.toString());
}
}
[email protected]可以看到toString返回的是Object的默認實現。
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
將生成的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
> 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運行即可