Наявність ланцюжка операцій "instanceof" вважається "запахом коду". Стандартна відповідь - «використовувати поліморфізм». Як би я це зробив у цьому випадку?
Існує ряд підкласів базового класу; жоден з них не під моїм контролем. Аналогічна ситуація була б з класами 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 і трохи більше коду можуть це виправити. Але "instanceof" здається настільки читабельним і доцільним у порівнянні. Чи є щось просте, але не смердюче?
Примітка додана 10.10.2010:
Виявляється, нові підкласи, ймовірно, будуть додані в майбутньому, і мій існуючий код доведеться грамотно обробляти їх. У цьому випадку HashMap для класу не буде працювати, оскільки клас не буде знайдений. Ланцюжок тверджень if, починаючи з найбільш конкретних і закінчуючи найзагальнішими, мабуть, найкращий зрештою:
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
}
[text](link)
для розміщення посилань у коментарях.