Java中是否存在类似Map的对象,用于存储和访问键/值对,但可以返回键的有序列表和值的有序列表,这样键和值列表的顺序是相同的?
因此,作为代码解释,我正在寻找一些行为类似于我虚构的OrderedMap的东西:
OrderedMap<Integer, String> om = new OrderedMap<>();
om.put(0, "Zero");
om.put(7, "Seven");
String o = om.get(7); // o is "Seven"
List<Integer> keys = om.getKeys();
List<String> values = om.getValues();
for(int i = 0; i < keys.size(); i++)
{
Integer key = keys.get(i);
String value = values.get(i);
Assert(om.get(key) == value);
}
我已经使用简单哈希映射,链表和集合排序映射的值。
import java.util.*;
import java.util.Map.*;
public class Solution {
public static void main(String[] args) {
// create a simple hash map and insert some key-value pairs into it
Map<String, Integer> map = new HashMap<String, Integer>();
map.put("Python", 3);
map.put("C", 0);
map.put("JavaScript", 4);
map.put("C++", 1);
map.put("Golang", 5);
map.put("Java", 2);
// Create a linked list from the above map entries
List<Entry<String, Integer>> list = new LinkedList<Entry<String, Integer>>(map.entrySet());
// sort the linked list using Collections.sort()
Collections.sort(list, new Comparator<Entry<String, Integer>>(){
@Override
public int compare(Entry<String, Integer> m1, Entry<String, Integer> m2) {
return m1.getValue().compareTo(m2.getValue());
}
});
for(Entry<String, Integer> value: list) {
System.out.println(value);
}
}
}
输出结果为:
C=0
C++=1
Java=2
Python=3
JavaScript=4
Golang=5
博士tl;
要保持Map< Integer, String >按键排序,使用实现SortedMap/NavigableMap接口的两个类之一:
TreeMap
ConcurrentSkipListMap
或者第三方实现。也许在谷歌番石榴或Eclipse集合(我没有检查)。
如果在单个线程中操作映射,请使用第一个TreeMap。如果跨线程操作,则使用第二个ConcurrentSkipListMap。
详细信息请参见下表和下面的讨论。
细节
下面是我制作的一个图形表,展示了与Java 11捆绑在一起的10个Map实现的特性。
NavigableMap接口是SortedMap的继承者。SortedMap逻辑上应该删除,但不能删除,因为一些第三方映射实现可能正在使用接口。
正如你在这个表中看到的,只有两个类实现了SortedMap/NavigableMap接口:
TreeMap
ConcurrentSkipListMap
这两种方法都按顺序保存键,或者按照它们的自然顺序(使用Comparable(https://docs.oracle.com/en/java/javase/11/docs/api/java.base/java/lang/Comparable.html)接口的compareTo方法),或者通过传递的Comparator实现。这两个类之间的区别在于第二个类ConcurrentSkipListMap是线程安全的、高度并发的。
另请参见下表中的迭代顺序列。
LinkedHashMap类按照条目最初被插入的顺序返回它们。
EnumMap按定义键的枚举类的顺序返回条目。例如,哪个员工负责一周中的哪一天的地图(map < DayOfWeek, Person >)使用Java中内置的DayOfWeek枚举类。该枚举定义为星期一的第一个和星期天的最后一个。所以迭代器中的条目会按照这个顺序出现。
其他六个实现没有承诺它们报告条目的顺序。
您可以利用NavigableMap接口,该接口可以按升序或降序键顺序访问和遍历。此接口旨在取代SortedMap接口。Navigable map通常根据它的键的自然排序,或者由创建map时提供的Comparator进行排序。
它有三种最有用的实现:TreeMap、ImmutableSortedMap和ConcurrentSkipListMap。
TreeMap例子:
TreeMap<String, Integer> users = new TreeMap<String, Integer>();
users.put("Bob", 1);
users.put("Alice", 2);
users.put("John", 3);
for (String key: users.keySet()) {
System.out.println(key + " (ID = "+ users.get(key) + ")");
}
输出:
Alice (ID = 2)
Bob (ID = 1)
John (ID = 3)