在
Java中,是否可以覆盖使用反射创建的类中的方法?例如,说我有以下类:
- 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方法,以便更改绑定的值.
解决方法
不,你的例子是不可能的.
在您的示例中,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();