在 Java中 ,是否可以覆盖使用创建的类中的方法reflection?例如,说我有以下课程:
reflection
public class MyObject { public String foo, bar; public MyObject(String foo) { this.foo = foo; this.bar = foo + "bar"; } public void setBar(String bar) { this.bar = bar; } }
在一个类中,我想直接创建它并重写其setBar方法,如下所示:
setBar
MyObject obj = new MyObject("something") { @Override public void setBar(String bar) { this.bar = this.foo; } };
有没有办法以同样的方式使用反射来覆盖方法?也许是这样的吗?:
Class<?> _class = Class.forName("com.example.MyObject"); Constructor<?> _constructor = _class.getConstructor(new Class<?>[]{String.class}); Method m = _class.getMethod("setBar", new Class<?>[]{String.class}); Object obj = _constructor.newInstance("Foo String") { m = new Method(new Class<?>[]{String.class}) { System.out.println("Foobar"); } };
如果没有,是否还有其他方法可以这样做,或者可以使用外部库吗?我正在寻找将侦听器添加到setter方法以更改绑定值的方法。
不,用您的例子是不可能的。
在您的示例中,Java编译器将创建两个单独的类:
MyObject.class MyObject$1.class
后者是使用覆盖方法的方法。在这种情况下,它是一个 匿名内部类 (请参阅Java教程文档)
但是,还有涉及字节码编织库的更复杂的解决方案。cglib,asm,javassist等库为您提供了一种在运行时动态创建新类并加载它们的工具。
Javassist提供了有关如何在运行时向类添加方法的教程。应该可以对其进行修改以添加/覆盖该方法,如下所示:
CtClass origClazz = ClassPool.getDefault().get("org.example.MyObject"); CtClass subClass = ClassPool.getDefault().makeClass(cls.getName() + "New", origClazz); CtMethod m = CtNewMethod.make( "public void setBar(String bar) { this.bar = bar; }", subClass ); subClass .addMethod(m); Class clazz = cc.toClass();