我正在尝试编写一个Java代理(用于动态检测),基于以下文章:http://blog.javabenchmark.org/2013/05/java-instrumentation-tutorial.html
问题是当我尝试使用代理时,我总是得到一个ClassNotFoundException(根据评论部分,其他跟随该文章的人似乎有同样的问题)。我的Java代理的代码如下:
Agent.java
package com.foo.tracer;
import java.lang.instrument.Instrumentation;
public class Agent {
public static void premain(String agentArgs, Instrumentation inst) {
inst.addTransformer(new MyClassFileTransformer());
}
}
MyClassTransformer.java:
package com.foo.tracer;
import java.lang.instrument.ClassFileTransformer;
import java.lang.instrument.IllegalClassFormatException;
import java.security.ProtectionDomain;
import javassist.*;
public class MyClassFileTransformer implements ClassFileTransformer {
public byte[] transform(ClassLoader loader, String className, Class classBeingRedefined,
ProtectionDomain protectionDomain, byte[] classfileBuffer) throws IllegalClassFormatException {
byte[] byteCode = classfileBuffer;
try {
// if I remove this line, everything works
ClassPool cp = ClassPool.getDefault();
} catch (Exception ex) {
ex.printStackTrace();
}
return byteCode;
}
}
我使用Maven(使用“mvn package”)构建项目:
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/maven-v4_0_0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>com.foo.tracer</groupId>
<artifactId>tracer</artifactId>
<packaging>jar</packaging>
<version>1.0-SNAPSHOT</version>
<name>tracer</name>
<url>http://maven.apache.org</url>
<build>
<plugins>
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-jar-plugin</artifactId>
<version>2.4</version>
<executions>
<execution>
<phase>process-classes</phase>
<goals>
<goal>jar</goal>
</goals>
<configuration>
<archive>
<manifestEntries>
<premain-class>com.foo.tracer.Agent</premain-class>
</manifestEntries>
</archive>
</configuration>
</execution>
</executions>
</plugin>
</plugins>
</build>
<dependencies>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>3.8.1</version>
<scope>test</scope>
</dependency>
<dependency>
<groupId>org.javassist</groupId>
<artifactId>javassist</artifactId>
<version>3.14.0-GA</version>
<type>jar</type>
</dependency>
</dependencies>
</project>
我按如下方式调用代理:
$ java -verbose:class -javaagent:PATH/tracer-1.0-SNAPSHOT.jar -jar my-app-1.0-SNAPSHOT.jar | grep -iE "(foo|javassist)"
[Loaded com.foo.tracer.Agent from file:[CUT]/tracer-1.0-SNAPSHOT.jar]
[Loaded com.foo.tracer.MyClassFileTransformer from file:[CUT]/tracer-1.0-SNAPSHOT.jar]
Exception in thread "main" java.lang.reflect.InvocationTargetException
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:498)
at sun.instrument.InstrumentationImpl.loadClassAndStartAgent(InstrumentationImpl.java:386)
at sun.instrument.InstrumentationImpl.loadClassAndCallPremain(InstrumentationImpl.java:401)
Caused by: java.lang.NoClassDefFoundError: javassist/ClassPool
at com.foo.tracer.Agent.premain(Agent.java:11)
... 6 more
Caused by: java.lang.ClassNotFoundException: javassist.ClassPool
at java.net.URLClassLoader.findClass(URLClassLoader.java:381)
at java.lang.ClassLoader.loadClass(ClassLoader.java:424)
at sun.misc.Launcher$AppClassLoader.loadClass(Launcher.java:338)
at java.lang.ClassLoader.loadClass(ClassLoader.java:357)
... 7 more
正如您所看到的,我自己的类由ClassLoader加载,但由于某种原因,javassist.jar中的.class文件未加载(我认为是javaassist.jar,它被打包到tracer-1.0-SNAPSHOT中。 jar,不由ClassLoader处理)。我猜我的pom.xml文件有问题。有人能告诉我如何解决这个问题吗?
目录结构(如果相关):
$ tree
.
├── pom.xml
└── src
├── javassist.jar
├── main
│ ├── java
│ │ └── com
│ │ └── foo
│ │ └── tracer
│ │ ├── Agent.java
│ │ └── MyClassFileTransformer.java
│ └── resources
│ └── javassist.jar
├── META-INF
│ └── MANIFEST.MF
└── test
└── java
└── com
└── tracer
└── AppTest.java
答案 0 :(得分:-1)
您假设javaassist.jar
中的文件正在打包到tracer-1.0-SNAPSHOT.jar
,但该假设不正确。
请参阅How can I create an executable JAR with dependencies using Maven?,了解有关如何实现这一目标的一些想法。