在 Java 中,是否可以覆盖使用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
方法,如下所示:
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方法以更改绑定值的方法。
答案 0 :(得分:11)
不,这不可能影响你的榜样。
在您的示例中,Java编译器将创建两个单独的类:
MyObject.class
MyObject$1.class
后者是具有重写方法的那个。在这种情况下,它是一个匿名内部类(参见Java tutorial documentation)
但是有更复杂的解决方案涉及字节码编织库。诸如cglib,asm,javassist等库可以为您提供在运行时动态创建新类并加载它们的工具。
Javassist有一个关于如何add methods to classes at runtime的教程。应该可以调整它来添加/覆盖方法,如下所示:
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();
答案 1 :(得分:2)
如果要返回接口类型的对象,可以使用Proxy.newProxyInstance
获取一个接口实例,该接口动态地将方法调用发送到InvocationHandler
对象,您可以编写该对象来执行任何操作你想要的自定义行为。
答案 2 :(得分:0)
不,你要的是运行时编译。虽然并非不可能,但它肯定不是反射API提供的东西。