- iOS/Objective-C 元类和类别
- objective-c - -1001 错误,当 NSURLSession 通过 httpproxy 和/etc/hosts
- java - 使用网络类获取 url 地址
- ios - 推送通知中不播放声音
我正在从 JAR
文件中读取一个类并注入(inject)一个函数。我如何将其写回 JAR 文件?
// Load the class representation
ClassPool pool = ClassPool.getDefault();
pool.insertClassPath( "c:/Test.jar" );
CtClass cc = pool.get("com.test.TestFunction");
CtMethod m = CtNewMethod.make("public void test2() { System.out.println(\"test2\"); }", cc);
cc.addMethod(m);
CtMethod cm = cc.getDeclaredMethod("test1", new CtClass[0]);
cm.insertBefore("{ test2();}");
cc.writeFile("c:/Test.jar"); // Fails here
线程“main”中的异常java.io.FileNotFoundException
:c:\Test.jar\com\test\TestFunction.class(系统找不到指定的路径)
最佳答案
我想 Javassist 中没有简单的方法可以更新 JAR 并将更新的类替换为新类。所以我创建了一个 JarHandler 类,它只接收参数。
这是执行注入(inject)的主类
public static void main(String args[]){
ClassPool pool = ClassPool.getDefault();
pool.insertClassPath( "c:/Test.jar" );
CtClass cc = pool.get("com.test.TestFunction");
CtMethod m = CtNewMethod.make("public void test2() { System.out.println(\"test2\"); }", cc);
cc.addMethod(m);
CtMethod cm = cc.getDeclaredMethod("test1", new CtClass[0]);
cm.insertBefore("{ test2();}");
byte[] b = cc.toBytecode(); // convert the new class to bytecode.
pool.removeClassPath(cp); // need to remove the classpath to release connection to JAR file so we can update it.
JarHandler jarHandler = new JarHandler();
jarHandler.replaceJarFile("C:/Test.jar", b, "com/test/TestFunction.class");
}
这是 JarHandler 类
public class JarHandler{
public void replaceJarFile(String jarPathAndName,byte[] fileByteCode,String fileName) throws IOException {
File jarFile = new File(jarPathAndName);
File tempJarFile = new File(jarPathAndName + ".tmp");
JarFile jar = new JarFile(jarFile);
boolean jarWasUpdated = false;
try {
=
JarOutputStream tempJar =
new JarOutputStream(new FileOutputStream(tempJarFile));
// Allocate a buffer for reading entry data.
byte[] buffer = new byte[1024];
int bytesRead;
try {
// Open the given file.
try {
// Create a jar entry and add it to the temp jar.
JarEntry entry = new JarEntry(fileName);
tempJar.putNextEntry(entry);
tempJar.write(fileByteCode);
} catch (Exception ex) {
System.out.println(ex);
// Add a stub entry here, so that the jar will close without an
// exception.
tempJar.putNextEntry(new JarEntry("stub"));
}
// Loop through the jar entries and add them to the temp jar,
// skipping the entry that was added to the temp jar already.
InputStream entryStream = null;
for (Enumeration entries = jar.entries(); entries.hasMoreElements(); ) {
// Get the next entry.
JarEntry entry = (JarEntry) entries.nextElement();
// If the entry has not been added already, so add it.
if (! entry.getName().equals(fileName)) {
// Get an input stream for the entry.
entryStream = jar.getInputStream(entry);
tempJar.putNextEntry(entry);
while ((bytesRead = entryStream.read(buffer)) != -1) {
tempJar.write(buffer, 0, bytesRead);
}
}else
System.out.println("Does Equal");
}
if(entryStream!=null)
entryStream.close();
jarWasUpdated = true;
}
catch (Exception ex) {
System.out.println(ex);
// IMportant so the jar will close without an
// exception.
tempJar.putNextEntry(new JarEntry("stub"));
}
finally {
tempJar.close();
}
}
finally {
jar.close();
if (! jarWasUpdated) {
tempJarFile.delete();
}
}
if (jarWasUpdated) {
if(jarFile.delete()){
tempJarFile.renameTo(jarFile);
System.out.println(jarPathAndName + " updated.");
}else
System.out.println("Could Not Delete JAR File");
}
}
这个函数只是通过将新的类字节码写入其中来创建一个临时 JAR。然后遍历当前 JAR 的所有条目并将其所有条目写入临时文件JAR 文件,除了正在更新的条目(字节码已经在上面写入)。然后它删除当前 JAR 并将其替换为使用相同 JAR 名称的临时 JAR。
关于Javassist - 如何将方法注入(inject) JAR 中的类,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/22591903/
Maven Artifact 之间有什么区别 javassist:javassist 和 org.javassist:javassist 在我的项目中,我的第一个依赖项是 3.8 版本,而我引入的第二
我有一个 CtMethod实例,但我不知道如何从中获取参数(不是类型)的名称。我试过 getParameterTypes ,但它似乎只返回类型。 我假设这是可能的,因为我使用的库没有源,只有类文件,我
可以使用 CtMethod.setBody("..") 设置方法主体,但我没有找到任何 API 来获取字符串格式的方法主体。 最佳答案 这不可能。 Javassist 不是反编译器。类文件中的方法以
可以使用CtMethod.setBody("..")设置方法体,但我没有找到任何API来获取字符串格式的方法体。 最佳答案 这是不可能的。 Javassist 不是反编译器。类文件中的方法用 Java
当使用 javassist 检测 com.sun.net.* 类时,我可以成功使用 .insertAfter 方法。但是,当尝试从 Glassfish 中运行的应用程序检测第三方类时,我收到错误: F
我正在使用 javassist 来更改方法体。当该方法在应用程序中定义时我可以做到这一点。但是,当我想更改应用程序使用的 jar 文件中定义的方法时,我会收到以下运行时错误: javassist.Ca
我在模型上使用 javassist:com.project.model.Model我已经尝试过各种组合如何为 ClassPath 和 CtClass 格式化字符串,但无济于事。 ClassPo
我正在处理 SpringMVC、Hibernate 和 JSON,但出现此错误。 HTTP Status 500 - Could not write JSON: No serializer found
我正在使用 javassist 创建一个类并为其添加注释。当我使用 CtClass.writeFile 并且我看到带有 Java 反编译器的类文件时,注释就在那里,但是当我使用 class.getAn
我想读取一个方法的返回值,并且我必须将其传递给使用 method.insertAfter 插入的代码。 示例: public String sayHello(){ return "1"; }
这是原始方法: @GET @Produces({"application/json"}) public Response getTermClouds(@Context SecurityCo
我正在尝试在应用程序启动期间使用附加功能来增强一些代码。整个设置本身工作正常,但有一点我认为 javassist 可能会生成错误的代码。 我正在特定类的特定方法上执行此操作,我之前检查过返回值实际上是
我正在编写一些 Javassist 代码来拦截方法调用并用代理替换它们。为此,我使用 ExprEditor 按以下方式替换调用: public static void main(String[] ar
我目前正在实现一个注释,强制字段通过 javassist 遵守条件。我想检查一个字段在读取时是否已初始化...因此,目前,我通过在虚拟机通过 Translator.onLoad(ClassPool p
我正在尝试使用 javassist 在编译时插入一些代码片段 环境 Java 8 Spring 启动 2.2.6 hibernate 5.4.12.Final Javasist nl.topicu
我对 javassist 有点问题,用方法处理程序装饰类。问题是方法处理程序与 Abc 类中的 method1 一起正常工作,但不会拦截对 Def 内部类中的 method2 的调用。 public
我将代码注入(inject)到 doGet 方法中 val replace = "" + " System.out.println(\"[before] " + className + "\")
我试图捕获分支和循环语句的条件表达式中涉及的变量值,例如: if (a + b < c - 5) { // here, capture value of a, b and c // if bo
我正在开发一个需要类检测的项目。我使用 javassist 因为在我的情况下进行检测非常方便。 我遇到了一个问题,可以使用以下代码片段进行描述: 假设 1 类: public class Class1
我试图在 java 类文件中找到类初始值设定项。我可以找到该方法,但如果在类文件中找不到 main,那么我希望它找到类初始值设定项并在那里注入(inject)代码。 如何使用 Javassist 找到
我是一名优秀的程序员,十分优秀!