使用反射,您可以很容易地实现这一点:
class EasyFieldAlterationAgent {
public static void premain(String args) throws Exception {
Field field = X.class.getDeclaredField("timeout");
field.setAccessible(true);
field.setValue(null, 42L); // set your value here.
}
}
请注意,这将更改的字段不是之前,而是就在类加载时间之后。如果这对您来说不可能,您也可以重新定义类本身,我只建议在以下情况下使用:
- 您的安全设置不允许您使用反射。
- 您需要在执行类的类型初始化程序之前更改该值。
如果您希望在加载类之前真正更改字段,那么您很幸运,您希望更改既是static 又定义原始值的字段的值。此类字段将其值直接存储在字段的位置。使用代理,您可以定义一个 ClassFileTransformer,它只是动态地更改字段的值。 ASM 是实现这种简单转换的好工具。使用此工具,您可以大致如下实现您的代理:
class FieldAlterationAgent {
public static void premain(String args, Instrumentation inst) {
instrumentation.addTransformer(new ClassFileTransformer() {
@Override
public void byte[] transform(ClassLoader loader,
String className,
Class<?> classBeingRedefined,
ProtectionDomain protectionDomain,
byte[] classfileBuffer)
throws IllegalClassFormatException {
if (!className.equals("X") {
return classFileBuffer;
}
ClassWriter classWriter = new ClassWriter(new ClassVisitor() {
@Override
public FieldVisitor visitField(int access,
String name,
String desc,
String signature,
Object value) {
if(name.equals("timeout") {
value = 42L; // set value here, make sure its a long!
}
return super.visitField(access, name, desc, signature, value);
}
}, 0);
new ClassReader(classFileBuffer).accept(classWriter);
return classWriter.toByteArray();
}
});
}
}
您可以看出后一个版本需要更多代码,并且需要将您的代理与其 ASM 依赖项打包在一起。
要应用代理,请将任一类放入jar file and put it onto the agent path。