我正在编写一些Javassist代码来拦截方法调用并用代理替换它们。为此,我使用 ExprEditor 以以下方式替换调用:
public static void main(String[] args) {
try {
ClassPool pool = ClassPool.getDefault();
CtClass ctClass = pool.get("Test");
CtMethod meth = ctClass.getDeclaredMethod("main");
meth.instrument(new ExprEditor() {
@Override
public void edit(final MethodCall m) throws CannotCompileException {
try {
if (m.getClassName().contains("Functions")) {
/* MAKE NON STATIC
CtClass old = pool.get(m.getClassName());
m.getMethod().setModifiers(Modifier.PUBLIC);
old.toClass();
*/
String s = m.getClassName() + " proxy = (" +
m.getClassName() + ") " + Main.class.getName() + ".create(" + m.getClassName() + ".class);" +
" $_ = proxy." + m.getMethodName() + "($$);";
m.replace(s);
}
} catch (Exception e) {
e.printStackTrace();
}
}
});
Class clazz = ctClass.toClass();
Method method = clazz.getMethod("main", String[].class);
method.invoke(null, new Object[]{new String[]{}});
} catch (InvocationTargetException e) {
e.getCause().printStackTrace();
} catch (NoSuchMethodException | IllegalAccessException | NotFoundException | CannotCompileException e) {
e.printStackTrace();
}
}
只要该方法不是静态的,就可以根据需要工作。现在我正在尝试通过使用注释代码将静态方法更改为非静态方法。在我看来,这应该有效,并且在 Javassist 文档中对其他修饰符也有类似的用法,但是当我取消注释并运行它时,我收到以下错误消息:
javassist.CannotCompileException: by java.lang.ClassFormatError: Arguments can't fit into locals in class file Functions/Color
我还尝试删除静态修饰符,而不仅仅是将修饰符设置为公共
m.getMethod().setModifiers(m.getMethod().getModifiers() & ~Modifier.STATIC);
但问题仍然存在。
这真的可能吗?
因此,您正在尝试从类 Test 中的保留入口点方法名称"main"中删除静态修饰符。我认为编译器不会让你这样做,因为 main 是一个保留的方法名称,只能有一个预定义的签名。而且,静态方法也是有问题的;当从类中调用时,如果删除 static 修饰符,则对它们的所有调用也会导致编译错误,因为它们不是原始代码中的实例方法。