Чтобы ответить на ваш вопрос в одном предложении:
По умолчанию у Карт нет последней записи, это не является частью их контракта.
И примечание: рекомендуется кодировать интерфейсы, а не классы реализации (см. Эффективная Java от Джошуа Блоха , глава 8, пункт 52: Ссылки на объекты по их интерфейсам ).
Итак, ваша декларация должна гласить:
Map<String,Integer> map = new HashMap<String,Integer>();
(Все карты имеют общий контракт, поэтому клиенту не нужно знать, что это за карта, если он не указывает подчиненный интерфейс с расширенным контрактом).
Возможные решения
Сортированные карты:
Существует подчиненный интерфейс SortedMap , который расширяет интерфейс карты с помощью методов поиска на основе порядка, и имеет подчиненный интерфейс NavigableMap , который расширяет его еще больше. Стандартная реализация этого интерфейса, TreeMap , позволяет сортировать записи либо путем естественного упорядочения (если они реализуют интерфейс Comparable ), либо с помощью прилагаемого Comparator .
Вы можете получить доступ к последней записи с помощью метода lastEntry :
NavigableMap<String,Integer> map = new TreeMap<String, Integer>();
// add some entries
Entry<String, Integer> lastEntry = map.lastEntry();
Связанные карты:
Существует также специальный случай LinkedHashMap , реализация HashMap, в которой хранится порядок вставки ключей. Однако нет интерфейса для резервного копирования этой функциональности, и нет прямого доступа к последнему ключу. Вы можете сделать это только с помощью таких хитростей, как использование списка между:
Map<String,String> map = new LinkedHashMap<String, Integer>();
// add some entries
List<Entry<String,Integer>> entryList =
new ArrayList<Map.Entry<String, Integer>>(map.entrySet());
Entry<String, Integer> lastEntry =
entryList.get(entryList.size()-1);
Правильное решение:
Поскольку вы не управляете порядком вставки, вам следует использовать интерфейс NavigableMap, т. Е. Вы бы написали компаратор, который помещает запись Not-Specified
последней.
Вот пример:
final NavigableMap<String,Integer> map =
new TreeMap<String, Integer>(new Comparator<String>() {
public int compare(final String o1, final String o2) {
int result;
if("Not-Specified".equals(o1)) {
result=1;
} else if("Not-Specified".equals(o2)) {
result=-1;
} else {
result =o1.compareTo(o2);
}
return result;
}
});
map.put("test", Integer.valueOf(2));
map.put("Not-Specified", Integer.valueOf(1));
map.put("testtest", Integer.valueOf(3));
final Entry<String, Integer> lastEntry = map.lastEntry();
System.out.println("Last key: "+lastEntry.getKey()
+ ", last value: "+lastEntry.getValue());
Выход:
Последний ключ: не указано, последнее значение: 1
Решение с использованием HashMap:
Если вы должны полагаться на HashMaps, все еще есть решение, использующее a) модифицированную версию вышеуказанного компаратора, b) List , инициализированный с помощью entrySet и c карты ) вспомогательный метод Collections.sort () :
final Map<String, Integer> map = new HashMap<String, Integer>();
map.put("test", Integer.valueOf(2));
map.put("Not-Specified", Integer.valueOf(1));
map.put("testtest", Integer.valueOf(3));
final List<Entry<String, Integer>> entries =
new ArrayList<Entry<String, Integer>>(map.entrySet());
Collections.sort(entries, new Comparator<Entry<String, Integer>>(){
public int compareKeys(final String o1, final String o2){
int result;
if("Not-Specified".equals(o1)){
result = 1;
} else if("Not-Specified".equals(o2)){
result = -1;
} else{
result = o1.compareTo(o2);
}
return result;
}
@Override
public int compare(final Entry<String, Integer> o1,
final Entry<String, Integer> o2){
return this.compareKeys(o1.getKey(), o2.getKey());
}
});
final Entry<String, Integer> lastEntry =
entries.get(entries.size() - 1);
System.out.println("Last key: " + lastEntry.getKey() + ", last value: "
+ lastEntry.getValue());
}
Выход:
Последний ключ: не указано, последнее значение: 1