Ответ 1
Чтобы ответить на ваш вопрос в одном предложении:
По умолчанию Карты не имеют последней записи, это не часть их контракта.
И примечание: хорошая практика кодировать интерфейсы, а не классы реализации (см. Эффективная Java от Джошуа Блоха, глава 8, Пункт 52: Обратитесь к объектам по их интерфейсам).
Итак, ваша декларация должна выглядеть следующим образом:
Map<String,Integer> map = new HashMap<String,Integer>();
(Все карты имеют общий контракт, поэтому клиенту не нужно знать, какая именно карта, если он не указывает дополнительный интерфейс с расширенным контрактом).
Возможные решения
Отсортированные карты:
Существует дополнительный интерфейс SortedMap, который расширяет интерфейс карты с помощью методов поиска на основе заказа и имеет дополнительный интерфейс NavigableMap, что еще больше расширяет его. Стандартная реализация этого интерфейса TreeMap позволяет сортировать записи либо путем естественного упорядочения (если они реализуют Сопоставимый интерфейс) или с помощью поставляемого 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, все еще существует решение, использующее: а) модифицированную версию вышеприведенного компаратора, b) a 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