Подтвердить что ты не робот

Как я могу переопределить методы в Java, когда создаю объект через отражение?

В 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, чтобы изменить привязанные значения.

4b9b3361

Ответ 1

Нет, это невозможно в способе вашего примера.

В вашем примере 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();

Ответ 2

Если вы возвращаете объект типа интерфейса, вы можете использовать Proxy.newProxyInstance, чтобы получить экземпляр интерфейса, который динамически отправляет вызовы метода в объект InvocationHandler, который вы можете написать для выполнения любого пользовательского поведения, которое вы хотите.

Ответ 3

Нет, то, что вы просите, похоже на компиляцию во время выполнения. Хотя это и не невозможно, это, безусловно, не то, что предоставляет API отражения.