получение свойства Java-бина неизвестного класса
Вопрос
Я хотел бы иметь возможность вызывать 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!");
}
}