14

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 方法以更改绑定值的方法。

4

3 回答 3

13

不,以您的示例方式不可能。

在您的示例中,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();
于 2012-12-08T19:48:06.063 回答
2

如果您要返回一个接口类型的对象,您可以使用Proxy.newProxyInstance该接口的实例来动态地将方法调用发送到一个InvocationHandler对象,您可以编写该对象来执行您想要的任何自定义行为。

于 2012-12-08T18:25:32.503 回答
0

不,您要求的是运行时编译之类的东西。虽然并非不可能,但它肯定不是反射 API 提供的东西。

于 2012-12-08T17:31:11.720 回答