Спустя примерно 20 лет программирование Java в первый раз, я хочу, чтобы у меня было множественное наследование. Но я не так, я ищу альтернативу для этой конкретной проблемы.
Реальное приложение - это своего рода ERP и несколько сложный, поэтому я пытаюсь перевести эту проблему на автомобили. Это происходит только до сих пор, но это лучшее, что я могу сделать.
Давайте начнем с интерфейса, описывающего, что может сделать автомобиль:
public interface Car {
public String accelerate();
public String decelerate();
public String steerLeft();
public String steerRight();
}
Теперь у нас есть базовая (но не абстрактная) реализация:
public class BasicCar implements Car {
protected final String name;
public BasicCar( String name ) {
this.name = name;
}
// In the real word this method is important and does lots of stuff like calculations and caching
protected String doDrive( String how ) {
return name + " is " + how + "ing";
}
@Override
public String accelerate() {
return doDrive( "accelerat" );
}
@Override
public String decelerate() {
return doDrive( "decelerat" );
}
// This method is important, too
protected String doSteer( String where ) {
return name + " is steering to the " + where;
}
@Override
public String steerLeft() {
return doSteer( "left" );
}
@Override
public String steerRight() {
return doSteer( "right" );
}
}
В реальном мире это само по себе является фасадом к DAO. Обратите внимание, что мне нужны методы doDrive
и doSteer
, потому что это то, где выполняется настоящая работа, например, некоторые вычисления, переводы и кеширование.
И еще несколько конкретных реализаций:
public class Sportscar extends BasicCar {
public Sportscar( String name ) {
super( name );
}
// I need to call the super method
@Override
public String doDrive( String how ) {
return super.doDrive( how ) + " fastly";
}
}
public class Truck extends BasicCar {
public Truck( String name ) {
super( name, new BasicMotor(), new TruckySteerer() );
}
// I need to call the super method
@Override
public String doSteer( String where ) {
return super.doSteer( where ) + " carefully";
}
}
Теперь я могу сделать следующее:
Car mcqueen = new Sportscar( "McQueen" );
mcqueen.steerLeft() //-> "McQueen is steering left"
mcqueen.accelerate() // -> "McQueen is accelerating fastly"
Car mack = new Truck( "Mack" );
mack.steerLeft() //-> "Mack is steering left carefully"
mack.accelerate() // -> "Mack is accelerating"
Теперь я хочу объединить эти два в один, который разделяет их функциональность:
Car red = new Firetruck( "Red" );
red.steerLeft() //-> "Red is steering left *carefully*"
red.accelerate() // -> "Red is accelerating *fastly*"
Что я пробовал/думал о
Я мог бы скопировать и вставить код из обоих в один класс. Но это никогда не бывает хорошей идеей. И в реальном приложении это довольно много кода, поэтому это еще худшая идея.
Я думаю, что я столкнулся с каким-то серьезным переписанием/рефакторингом.
Поэтому я мог бы сделать Sportscar
и Truck
Decorator, а для Firetruck
использовать оба. Но это не сработает, потому что doSteer
и doDrive
вызываются из украшенных объектов, тогда как декоратор работает только для вызовов из "снаружи". Поэтому мне пришлось бы поместить эти методы в декоратор, тоже, и это тоже не очень хорошая идея.
Поэтому я мог бы использовать новые причудливые функции Java8 и сделать doSteer
и doDrive
делегирование интерфейса, например:
@FunctionalInterface
interface Driver {
public String doDrive( String where );
}
И подавая его конструктору BasicCar
но тогда (кроме того, что вы становитесь довольно сложными и получаете некоторые неприятные проблемы с java с final
), я больше не имею доступа к состоянию BasicCar
.
Так что в настоящее время я немного потерялся здесь. Любые хорошие идеи будут рассмотрены.
EDIT: Чтобы привести пример, как это выглядит в реальном мире: группа doSteer
может быть примерно такой:
class ProductImpl {
String getTitle(){
return getField( "title" );
}
String getTeaser(){
return getField( "teaser" );
}
String getField( String name ){
Locale loc = configuredLocale();
Map vars = configuredVarialbes();
String value = getCached( name, loc );
value = translateVariables( value );
value = replaceVariables( value, vars );
// and more
}
}