Хорошая комбинация использования интерфейса и дженериков, безусловно, приведет к простой реализации чистых, которая также расширяема.
Вот что я предлагаю - Я на самом деле реализовать это, и это работает прекрасно:
Обновление на основе комментариев от OP
Интерфейс Модель может быть «обобщен», чтобы ограничить типы возвращаемых:
package org.example;
/**
* Created by prahaladd on 08/07/15.
*/
public interface Model<T extends Identifier>
{
T getIdentifier();
}
Реализовать класс модели, которая использует конкретный тип идентификатора:
package org.example;
/**
* Created by prahaladd on 08/07/15.
*/
public class Person implements Model<StringIdentifier>
{
private final String name;
private final String id;
public Person(String id, String name)
{
this.id = id;
this.name = name;
}
@Override
public StringIdentifier getIdentifier()
{
return new StringIdentifier(id);
}
public String getName()
{
return name;
}
}
Внедрение ChangeSet теперь немного изменяет для имитации интерфейса карты, как показано ниже. Это в-то, в настоящее время занимает в типе идентификаторов, которые будут храниться в качестве ключей:
package org.example;
import java.util.Map;
/**
* Created by prahaladd on 08/07/15.
*/
public class ChangeSet<T extends Identifier, M extends Model<T>>
{
//Refer to PECS - http://stackoverflow.com/questions/2723397/java-generics-what-is-pecs
private Map<? super Identifier, M> changeMap;
public void addChange(M element)
{
changeMap.put(element.getIdentifier(),element);
}
public M getChangedElementForId(T id)
{
return changeMap.get(id);
}
}
Все эти изменения не так уж плохо - вы можете создать экземпляр ревизии, в значительной степени легко, как показано ниже:
package org.example;
public class Main {
public static void main(String[] args)
{
Person p1 = new Person("1", "Tom");
Person p2 = new Person("2", "Jerry");
//change set is instantiated without any redundant generic parameters
ChangeSet<StringIdentifier, Person> changes = new ChangeSet<StringIdentifier,Person>();
//assume that there were some changes and you want to add them to the changeset.
changes.addChange(p1);
changes.addChange(p2);
//retrieve element from the changeset for an id
p1= changes.getChangedElementForId(new StringIdentifier("1"));
p2 = changes.getChangedElementForId(new StringIdentifier("2"));
}
}
Альтернативное решение
Во-первых - определить интерфейс, который инкапсулирует идентификатор. Это не перебор; при условии, что у вас есть различные типы идентификаторов с помощью интерфейса, чтобы определить контракт на идентификатор будет идти долгий путь, чтобы сделать ваш код чистым и расширяемым:
package org.example;
/**
* Created by prahaladd on 08/07/15.
*/
public interface Identifier<T>
{
T getIdentifier();
}
Теперь, когда вы определили интерфейс идентификатора, вы можете определить различные реализации для него соответствуют вашим различным типам идентификаторов. Напр. Ниже я обеспечил реализацию для StringIdentifier, который генерирует идентификаторы типа строки:
package org.example;
/**
* Created by prahaladd on 08/07/15.
*/
public class StringIdentifier implements Identifier<String>
{
private final String identifier;
public StringIdentifier(String id)
{
identifier = id;
}
@Override
public String getIdentifier()
{
return "someId";
}
}
Теперь определим интерфейс модели. В идеале интерфейс модели не должен иметь дело с каким-либо идентификационным типом, он должен просто знать, что он должен вернуть идентификатор (как и ваш прецедент).
package org.example;
/**
* Created by prahaladd on 08/07/15.
*/
public interface Model
{
Identifier getIdentifier();
}
Теперь предоставьте реализацию интерфейса модели. Напр. ниже - класс Person, упомянутый в вашем запросе:
package org.example;
/**
* Created by prahaladd on 08/07/15.
*/
public class Person implements Model
{
private final String name;
private final String id;
public Person(String id, String name)
{
this.id = id;
this.name = name;
}
@Override
public Identifier getIdentifier()
{
return new StringIdentifier(id);
}
public String getName()
{
return name;
}
}
Теперь определите ChangeSet. ChangeSet должен знать только, что он хранит сопоставление между объектами ID и соответствующей моделью. Он действительно не знает о типе объектов ID. Это делает класс ChangeSet чрезвычайно гибким, чтобы даже поддерживать гетерогенную коллекцию в дополнение к однородным, которые вы хотите.
package org.example;
import java.util.Map;
/**
* Created by prahaladd on 08/07/15.
*/
public class ChangeSet<M extends Model>
{
//Refer to PECS - http://stackoverflow.com/questions/2723397/java-generics-what-is-pecs
private Map<? super Identifier, M> changeMap;
private Class identifierType;
public void addChange(M element)
{
//prahaladd - update : save the identifier type for a later check.
if(identifierType != null)
{
identifierType = element.getIdentifier.getClass();
}
changeMap.put(element.getIdentifier(),element);
}
public M getChangedElementForId(Identifier id)
{
//prahaladd updated - verify that the type of the passed in id
//is the same as that of the changeset identifier type.
if(!id.getClass().equals(identifierType))
{
throw new IllegalArgumentException();
}
return changeMap.get(id);
}
}
Теперь тяжелая работа окупается. Взгляните на приведенную ниже клиентскую реализацию:
package org.example;
public class Main {
public static void main(String[] args)
{
Person p1 = new Person("1", "Tom");
Person p2 = new Person("2", "Jerry");
ChangeSet<Person> changes = new ChangeSet<Person>();
//assume that there were some changes and you want to add them to the changeset.
changes.addChange(p1);
changes.addChange(p2);
//retrieve element from the changeset for an id
p1= changes.getChangedElementForId(new StringIdentifier("1"));
p2 = changes.getChangedElementForId(new StringIdentifier("2"));
}
}
Точно, как вы предполагали! Как вы можете видеть, нет ничего необычного в том, что здесь было сделано. Простые объектно-ориентированные концепции и продуманная комбинация интерфейса и дженериков.
Надеюсь, что это поможет!
какая версия JDK вы используете? –
Я использую JDK 1.7 – JSBach
С 1.7, я думаю, что это самое ближайшее, что мы можем получить. В Java 8 вы сможете записать это как «ChangeSet changeSet = new ChangeSet ();' без дублирования. Который может быть далее записан как «ChangeSet changeSet = new ChangeSet <>();' с алмазным оператором. Я думаю, что это возможно с помощью ввода типа - https://docs.oracle.com/javase/tutorial/java/generics/genTypeInference.html –