получение свойства Java-бина неизвестного класса

StackOverflow https://stackoverflow.com/questions/483980

  •  20-08-2019
  •  | 
  •  

Вопрос

Я хотел бы иметь возможность вызывать getProgram для объектов, имеющих этот метод, не зная, к какому классу они принадлежат.Я знаю, что здесь мне следует использовать интерфейс, но я работаю с чужим кодом и не могу перепроектировать классы, с которыми работаю.Я думал, что BeanUtils.getProperty может мне помочь, но, похоже, он возвращает только строки.Есть ли что-то вроде Beanutils.getProperty, которое вернет объект, допускающий приведение типов?Или другой, более разумный способ работы с двумя похожими классами, не имеющими общего интерфейса?Спасибо, -Морган

Это было полезно?

Решение

Предположительно, у вас есть конечное число классов, реализующих этот метод, и вы можете ссылаться на них напрямую.Так что вам не нужна рефлексия.Отражение – это зло.

Предположим, у вас есть набор классов с методом:

public class LibA { public Program getProgram() { return program; } ... };
public class LibB { public Program getProgram() { return program; } ... };
...

Тогда вам просто нужны пары экземпляров/приведения.Вы можете поместить это в метод, чтобы вам нужно было сделать это только один раз.

public static Program getProgram(Object obj) {
    if        (obj instanceof LibA) {
        return              ((LibA)obj).getProgram();
    } else if (obj instanceof LibB) {
        return              ((LibB)obj).getProgram();
    } else {
        throw new IllegalArgumentException(obj+" doesn't have a known getProgram");
            // Or an appropriate application exception.
    }
}

В качестве альтернативы вы можете использовать адаптер:

public interface ProgramContainer { 
    Program getProgram();
    ...
}

public class LibAContainer implements ProgramContainer {
    private final LibA libA;
    public LibAContainer(LibA libA) {
        this.libA = libA;
    }
    public Program getProgram() {
        return libA.getProgram();
    }
    ...
}

Другие советы

Используйте PropertyUtils (из apache commons-beanutils) вместо BeanUtils.

У него есть метод getProperty(Object bean, String name), который возвращает объект вместо строки.

См. JavaDoc Чтобы получить больше информации.

Просто используйте для этого отражение...В следующем примере показано, как это сделать с объектами, не имеющими общего интерфейса.

    public static void main(String[] args) throws Exception {
    doSomething(new A());
    doSomething(new B());
}

private static void doSomething(Object object) throws Exception {
    Method m = object.getClass().getMethod("doSomething", (Class[])null);
    m.invoke(object, (Object[])null);
}

private static class A {
    public void doSomething() {
        System.out.println("I'm doing it already!");
    }
}

private static class B {
    public void doSomething() {
        System.out.println("I'm doing it too!");
    }
}

См. API отражения:

Используйте Class.getMethod() (или getMethods()), чтобы найти подходящий метод и вызвать его.

Довольно простое решение:Используйте делегирование, которое реализует интерфейс:

interface GetProgram
{
    String getProgram ();
}

class AWrapper implements GetProgram
{
    A a;
    public AWrapper (A a) { this.a = a;
    String getProgram () { return a.getProgram(); }
}

Теперь вы можете использовать интерфейс в своем коде, не затрагивая исходные классы.

Недостаток:Это не сработает, если вы А созданы где-то вне вашей досягаемости.Лучше всего работает, если A создается один раз под вашим контролем, и вы можете немедленно его обернуть.

Немного более короткая версия, если у вас Java 5+.

public static void main(String[] args) throws Exception {
    System.out.println(invoke("toString", new A());
    System.out.println(invoke("toString", new B());
}

private static <R> R invoke(Object object, String methodName) throws Exception {
    return (R) object.getClass().getMethod(methodName).invoke(object);
}

java.beans.Выражение сделает это, если метод доступен в конкретном классе получателя.

public static void main(String[] args) throws Exception {
    new Expression(new A(), "doSomething", null).getValue();
    new Expression(new B(), "doSomething", null).getValue();
}

public static class A {
    public void doSomething() {
            System.out.println("I'm doing it already!");
    }
}

public static class B {
    public void doSomething() {
            System.out.println("I'm doing it too!");
    }
}
Лицензировано под: CC-BY-SA с атрибуция
Не связан с StackOverflow
scroll top