java - 为什么我的 Javassist 类没有在运行时执行?

标签 java maven jar javassist

我一直在编写一个java代理来修改加载类的字节码。 我的代码由 4 个类组成:

  1. StackTraceAgent.java - 带有方法的java代理类 普雷梅因。
  2. StackTraceClassTransformer.java - 实现 java.lang.instrument.ClassFileTransformer。
  3. Student.java - 只是用于测试的自定义类。
  4. AgentTest - 具有 main 方法的简单类。

使用的库:

  1. javassist-3.20.0-GA.jar

StackTraceAgent.java、ClassFileTransformer.java 和 javassist-3.20.0-GA.jar 打包在 jar 文件中。它是通过“-javaagent:MyJar.jar AgentTest”调用的。我的问题是使用 javassist 类编写的代码在运行时不会被调用。它只是被运行时传递出去,没有任何错误,最后程序继续执行非 javassist 代码直到结束。

我如何尝试解决这个问题:

  1. 我已经复习了编译的类(class)。所有类都包含精确的代码。
  2. 我感觉打包成jar有问题,但是经过几个小时的上网我没有发现任何有趣的东西。

list .MF:

Manifest-Version: 1.0
Premain-Class: agent.StackTraceAgent

我的 pom.xml 文件:

    <?xml version="1.0" encoding="UTF-8"?>
<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/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>

    <groupId>com.belenov</groupId>
    <artifactId>TraceAgent</artifactId>
    <version>1.0-SNAPSHOT</version>

    <dependencies>
        <!-- https://mvnrepository.com/artifact/org.javassist/javassist -->
        <dependency>
            <groupId>org.javassist</groupId>
            <artifactId>javassist</artifactId>
            <version>3.20.0-GA</version>
        </dependency>

    </dependencies>

    <build>
        <plugins>

            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-dependency-plugin</artifactId>
                <executions>
                    <execution>
                        <id>copy-dependencies</id>
                        <phase>prepare-package</phase>
                        <goals>
                            <goal>copy-dependencies</goal>
                        </goals>
                        <configuration>
                            <outputDirectory>${project.build.directory}/classes/lib</outputDirectory>
                            <overWriteReleases>false</overWriteReleases>
                            <overWriteSnapshots>false</overWriteSnapshots>
                            <overWriteIfNewer>true</overWriteIfNewer>
                        </configuration>
                    </execution>
                </executions>
            </plugin>

            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-jar-plugin</artifactId>
                <configuration>
                    <archive>
                        <manifestFile>
                            src/main/resources/META-INF/MANIFEST.MF
                        </manifestFile>
                        <manifest>
                            <addClasspath>false</addClasspath>
                        </manifest>
                    </archive>
                </configuration>
            </plugin>
        </plugins>
    </build>
</project>

StackTraceAgent.java:

import java.lang.instrument.Instrumentation;
public class StackTraceAgent  {

    public static void premain(String args, Instrumentation instrumentation) {
        System.out.println("instrument agent");
        StackTraceClassTransformer transformer = new StackTraceClassTransformer();
        instrumentation.addTransformer(transformer);
    }
}

StackTraceClassTransformer.java:

import javassist.ClassPool;
import javassist.CtClass;
import javassist.CtMethod;

import java.lang.instrument.ClassFileTransformer;
import java.lang.instrument.IllegalClassFormatException;
import java.security.ProtectionDomain;

public class StackTraceClassTransformer implements ClassFileTransformer {

    public byte[] transform(ClassLoader loader, String className, Class<?> classBeingRedefined,
                            ProtectionDomain protectionDomain,
                            byte[] classfileBuffer) throws IllegalClassFormatException {

        try {
            //the code below is passed up
            ClassPool classPool = ClassPool.getDefault();
            CtClass ctClass = classPool.get(className);
            CtMethod[] methods = ctClass.getMethods();

            for (CtMethod method : methods) {
                method.insertAfter("System.out.println(\"Method name: \"+" + method.getName() + "+\" \"+" + method.getSignature() + ");");
                method.insertAfter("System.out.println(\"Object: \" + this.toString());");
            }

            byte[] byteCode = ctClass.toBytecode();
            ctClass.detach();
            return byteCode;
        } catch (Exception ex) {
            ex.printStackTrace();
        }
        return null;
    }
}

AgentTest.java

public class AgentTest {
    public static void main(String[] args) {
        System.out.println("Agent Tester");
        Student student = new Student("vasia", "poopkin");
        student.toString();
    }
}

学生.java

public class Student {
    private String name;
    private String surname;

    public Student(String name, String surname) {
        this.name = name;
        this.surname = surname;
    }

    @Override
    public String toString() {
        return "Student{" +
                "name='" + name + '\'' +
                ", surname='" + surname + '\'' +
                '}';
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public String getSurname() {
        return surname;
    }

    public void setSurname(String surname) {
        this.surname = surname;
    }
}

最佳答案

基本上在 list 文件中,您需要包含程序使用的每个库和资源的类路径,您可以在 MANIFEST.MF 中添加如下行:

Class-Path: path/to/my/libraries

一种方法是简单地更新 pom.xml,即 Maven jar 插件的 list 部分:

<addClasspath>true</addClasspath>
<classpathPrefix>${project.build.directory}/classes/lib</classpathPrefix>

关于java - 为什么我的 Javassist 类没有在运行时执行?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/42483463/

相关文章:

java - 当 validator 发现表单错误时,表单页面将在 POST url 处重新显示

eclipse - 如何为 m2eclipse 类型的项目使用自定义生命周期?

java - Maven 中的集成测试问题

java - 使用 Maven 创建一个包含已编译类和 javadoc 的 jar

java - Java中的方法与构造函数

java - 如何将文件发送给客户端而不立即保存

Java NoClassDefFoundError

java - 在另一台主机上部署jar时没有音频/图像

java - 可运行的 .jar 可在 Ubuntu 上运行,但不能在 Windows 上运行

java - gradle 和 junit 间歇性并发故障