Наличие цепочки операций «instanceof» считается «запахом кода». Стандартный ответ - «полиморфизм использования». Как мне это сделать в этом случае?Избегание instanceof в Java
Существует ряд подклассов базового класса; ни один из них не находится под моим контролем. Аналогичная ситуация была бы с классами Java Integer, Double, BigDecimal и т.д.
if (obj instanceof Integer) {NumberStuff.handle((Integer)obj);}
else if (obj instanceof BigDecimal) {BigDecimalStuff.handle((BigDecimal)obj);}
else if (obj instanceof Double) {DoubleStuff.handle((Double)obj);}
У меня есть контроль над NumberStuff и так далее.
Я не хочу использовать много строк кода, где бы выполнялись несколько строк. (Иногда я делаю отображение HashMap Integer.class к экземпляру IntegerStuff, BigDecimal.class к экземпляру BigDecimalStuff и т.д. Но сегодня я хочу что-то попроще.)
Я хотел бы что-то же просто, как это:
public static handle(Integer num) { ... }
public static handle(BigDecimal num) { ... }
Но Java просто не работает таким образом.
Я хотел бы использовать статические методы при форматировании. То, что я форматирование, является составным, где Thing1 может содержать массив Thing2s, а Thing2 может содержать массив Thing1s. У меня была проблема, когда я реализовал свои форматтеры так:
class Thing1Formatter {
private static Thing2Formatter thing2Formatter = new Thing2Formatter();
public format(Thing thing) {
thing2Formatter.format(thing.innerThing2);
}
}
class Thing2Formatter {
private static Thing1Formatter thing1Formatter = new Thing1Formatter();
public format(Thing2 thing) {
thing1Formatter.format(thing.innerThing1);
}
}
Да, я знаю, что HashMap и немного больше кода может исправить это. Но «экземпляр» кажется таким читабельным и удобным для сравнения. Есть что-то простое, но не вонючее?
Примечание добавлено 5/10/2010:
Оказывается, что новые подклассы, вероятно, будут добавлены в будущем, и мой существующий код придется обращаться с ними корректно. В этом случае HashMap on Class не будет работать, потому что класс не будет найден. Цепь, если заявления, начиная с наиболее специфичны и заканчивая самым общим, является, пожалуй, лучше всего после того, как все:
if (obj instanceof SubClass1) {
// Handle all the methods and properties of SubClass1
} else if (obj instanceof SubClass2) {
// Handle all the methods and properties of SubClass2
} else if (obj instanceof Interface3) {
// Unknown class but it implements Interface3
// so handle those methods and properties
} else if (obj instanceof Interface4) {
// likewise. May want to also handle case of
// object that implements both interfaces.
} else {
// New (unknown) subclass; do what I can with the base class
}
Я предлагаю [образец посетителя] [1]. [1]: http://en.wikipedia.org/wiki/Visitor_pattern – lexicore
Является ли это цепью ifthen, против которой вы возражаете, или просто использованием «instanceof»? – Greg
Шаблон посетителя требует добавления метода к целевому классу (например, Integer) - простой в JavaScript, жесткий на Java. Отличная модель при проектировании целевых классов; не так просто, пытаясь научить старый класс новым трюкам. –