java map笔记

java map笔记

第一种方法根据键值的名字取值

import java.util.HashMap;
import java.util.Map;

public class Test {

 /**
  * @param args
  */
 public static void main(String[] args) {
  // TODO Auto-generated method stub
        Map<String, Object> map=new HashMap<String, Object>();
        map.put("id", "sunny");
        map.put("value", "syl");
        
        System.out.println(map.get("id"));

        System.out.println(map.get("value"));
 }

}

第二种方法 遍历

import java.util.HashMap;
import java.util.Map;
import java.util.Set;

public class Test {

 /**
  * @param args
  */
 public static void main(String[] args) {
  // TODO Auto-generated method stub
        Map<String, Object> map=new HashMap<String, Object>();
        map.put("id", "sunny");
        map.put("value", "syl");
        
        Set<String> get = map.keySet(); 
        for (String test:get) {
         System.out.println(test+","+map.get(test));
        }
 }

}

 

 

 

java 遍历Map 和 根据Map的值(value)取键(key)

1. public static void main(String[] args) {  

2. //      Map<String, Object> map = new HashMap<String, Object>();  

3.         Map<String, Object> map = new LinkedHashMap<String, Object>();  

4.         map.put("username","zhaokuo");  

5.         map.put("password""123456");  

6.         map.put("email""zhaokuo719@163.com");  

7.         map.put("sex""男");  

8.           

9.         //第一种 用for循环的方式  

10.         for (Map.Entry<String, Object> m :map.entrySet())  {  

11.             System.out.println(m.getKey()+"\t"+m.getValue());  

12.         }  

13.           

14.         //利用迭代 (Iterator)  

15.         Set set=map.entrySet();  

16.         Iterator iterator=set.iterator();  

17.         while(iterator.hasNext()){  

18.                 Map.Entry<String, Object> enter=(Entry<String, Object>) iterator.next();  

19.                 System.out.println(enter.getKey()+"\t"+enter.getValue());  

20.         }  

21.         //利用KeySet 迭代  

22.         Iterator it = map.keySet().iterator();  

23.         while(it.hasNext()){  

24.              String key;     

25.              String value;     

26.              key=it.next().toString();     

27.              value=(String) map.get(key);     

28.              System.out.println(key+"--"+value);     

29.         }  

30.         //利用EnterySet迭代  

31.         Iterator i=map.entrySet().iterator();            

32.         System.out.println( map.entrySet().size());     

33.         String key;            

34.         String value;     

35.         while(i.hasNext()){     

36.                 Map.Entry entry = (Map.Entry)i.next();            

37.                 key=entry.getKey().toString();            

38.                 value=entry.getValue().toString();            

39.                 System.out.println(key+"===="+value);                      

40.         }    

41.           

42.         System.out.println(getKeyByValue(map, "zhaokuo"));  

43.     }  

1. //根据Value取Key  

2.     public static String getKeyByValue(Map map, Object value) {  

3.         String keys="";  

4.         Iterator it = map.entrySet().iterator();  

5.         while (it.hasNext()) {  

6.             Map.Entry entry = (Entry) it.next();  

7.             Object obj = entry.getValue();  

8.             if (obj != null && obj.equals(value)) {  

9.                 keys=(String) entry.getKey();  

10.             }  

11.   

12.   

13.         }  

14.         return keys;  

15.         }  

Java遍历方式

1. 阐述

  对于Java中Map的遍历方式,很多文章都推荐使用entrySet,认为其比keySet的效率高很多。理由是:entrySet方法一次拿到所有key和value的集合;而keySet拿到的只是key的集合,针对每个key,都要去Map中额外查找一次value,从而降低了总体效率。那么实际情况如何呢?

  为了解遍历性能的真实差距,包括在遍历key+value、遍历key、遍历value等不同场景下的差异,我试着进行了一些对比测试。

2. 对比测试

  一开始只进行了简单的测试,但结果却表明keySet的性能更好,这一点让我很是费解,不都说entrySet明显好于keySet吗?为了进一步地进行验证,于是采用了不同的测试数据进行更详细的对比测试。

2.1 测试数据

2.1.1 HashMap测试数据

· HashMap-1,大小为100万,key和value均为String,key的值为1、2、3……1000000:

Map<String, String> map = new HashMap<String, String>();

String key, value;

for (i = 1; i <= num; i++) {

    key = "" + i;

    value = "value";

    map.put(key, value);

}

· HashMap-2,大小为100万,key和value均为String,key的值为50、100、150、200、……、50000000:

Map<String, String> map = new HashMap<String, String>();

String key, value;

for (i = 1; i <= num; i++) {

    key = "" + (i * 50);

    value = "value";

    map.put(key, value);

}

2.1.2 TreeMap测试数据

· TreeMap-1,大小为100万,key和value均为String,key的值为1、2、3……1000000:

Map<String, String> map = new TreeMap<String, String>();

String key, value;

for (i = 1; i <= num; i++) {

    key = "" + i;

    value = "value";

    map.put(key, value);

}

· TreeMap-2,大小为100万,key和value均为String,key的值为50、100、150、200、……、50000000,更离散:

Map<String, String> map = new TreeMap<String, String>();

String key, value;

for (i = 1; i <= num; i++) {

    key = "" + (i * 50);

    value = "value";

    map.put(key, value);

}

2.2 测试场景

  分别使用keySet、entrySet和values的多种写法测试三种场景:遍历key+value、遍历key、遍历value的场景。

2.2.1 遍历key+value

· keySet遍历key+value(写法1):

Iterator<String> iter = map.keySet().iterator();

while (iter.hasNext()) {

    key = iter.next();

    value = map.get(key);

}

· keySet遍历key+value(写法2):

for (String key : map.keySet()) {

    value = map.get(key);

}

· entrySet遍历key+value(写法1):

Iterator<Entry<String, String>> iter = map.entrySet().iterator();

Entry<String, String> entry;

while (iter.hasNext()) {

    entry = iter.next();

    key = entry.getKey();

    value = entry.getValue();

}

·  entrySet遍历key+value(写法2):

for (Entry<String, String> entry: map.entrySet()) {

    key = entry.getKey();

    value = entry.getValue();

}

2.2.2 遍历key

· keySet遍历key(写法1):

Iterator<String> iter = map.keySet().iterator();

while (iter.hasNext()) {

    key = iter.next();

}

· keySet遍历key(写法2):

for (String key : map.keySet()) {

}

·  entrySet遍历key(写法1):

Iterator<Entry<String, String>> iter = map.entrySet().iterator();

while (iter.hasNext()) {

    key = iter.next().getKey();

}

· entrySet遍历key(写法2):

for (Entry<String, String> entry: map.entrySet()) {

    key = entry.getKey();

}

2.2.3 遍历value

· keySet遍历value(写法1):

Iterator<String> iter = map.keySet().iterator();

while (iter.hasNext()) {

    value = map.get(iter.next());

}

· keySet遍历value(写法2):

for (String key : map.keySet()) {

    value = map.get(key);

}

· entrySet遍历value(写法1):

Iterator<Entry<String, String>> iter = map.entrySet().iterator();

while (iter.hasNext()) {

value = iter.next().getValue();

}

· entrySet遍历value(写法2):

for (Entry<String, String> entry: map.entrySet()) {

    value = entry.getValue();

}

· values遍历value(写法1):

Iterator<String> iter = map.values().iterator();

while (iter.hasNext()) {

value = iter.next();

}

· values遍历value(写法2):

for (String value : map.values()) {

}

2.3 测试结果

2.3.1 HashMap测试结果

单位:毫秒

HashMap-1

HashMap-2

keySet遍历key+value(写法1)

39

93

keySet遍历key+value(写法2)

38

87

entrySet遍历key+value(写法1)

43

86

entrySet遍历key+value(写法2)

43

85

 

单位:毫秒

HashMap-1

HashMap-2

keySet遍历key(写法1)

27

65

keySet遍历key(写法2)

26

64

entrySet遍历key(写法1)

35

75

entrySet遍历key(写法2)

34

74

 

单位:毫秒

HashMap-1

HashMap-2

keySet遍历value(写法1)

38

87

keySet遍历value(写法2)

37

87

entrySet遍历value(写法1)

34

61

entrySet遍历value(写法2)

32

62

values遍历value(写法1)

26

48

values遍历value(写法2)

26

48

2.3.2 TreeMap测试结果

单位:毫秒

TreeMap-1

TreeMap-2

keySet遍历key+value(写法1)

430

451

keySet遍历key+value(写法2)

429

450

entrySet遍历key+value(写法1)

77

84

entrySet遍历key+value(写法2)

70

68

 

单位:毫秒

TreeMap-1

TreeMap-2

keySet遍历key(写法1)

50

49

keySet遍历key(写法2)

49

48

entrySet遍历key(写法1)

66

64

entrySet遍历key(写法2)

65

63

 

单位:毫秒

TreeMap-1

TreeMap-2

keySet遍历value(写法1)

432

448

keySet遍历value(写法2)

430

448

entrySet遍历value(写法1)

62

61

entrySet遍历value(写法2)

62

61

values遍历value(写法1)

46

46

values遍历value(写法2)

45

46

3. 结论

3.1 如果你使用HashMap

1. 同时遍历key和value时,keySet与entrySet方法的性能差异取决于key的具体情况,如复杂度(复杂对象)、离散度、冲突率等。换言之,取决于HashMap查找value的开销。entrySet一次性取出所有key和value的操作是有性能开销的,当这个损失小于HashMap查找value的开销时,entrySet的性能优势就会体现出来。例如上述对比测试中,当key是最简单的数值字符串时,keySet可能反而会更高效,耗时比entrySet少10%。总体来说还是推荐使用entrySet。因为当key很简单时,其性能或许会略低于keySet,但却是可控的;而随着key的复杂化,entrySet的优势将会明显体现出来。当然,我们可以根据实际情况进行选择

2. 只遍历key时,keySet方法更为合适,因为entrySet将无用的value也给取出来了,浪费了性能和空间。在上述测试结果中,keySet比entrySet方法耗时少23%。

3. 只遍历value时,使用vlaues方法是最佳选择,entrySet会略好于keySet方法。

4. 在不同的遍历写法中,推荐使用如下写法,其效率略高一些:

for (String key : map.keySet()) {

    value = map.get(key);

}

 

for (Entry<String, String> entry: map.entrySet()) {

    key = entry.getKey();

    value = entry.getValue();

}

 

for (String value : map.values()) {

}

3.2 如果你使用TreeMap

1. 同时遍历key和value时,与HashMap不同,entrySet的性能远远高于keySet。这是由TreeMap的查询效率决定的,也就是说,TreeMap查找value的开销较大,明显高于entrySet一次性取出所有key和value的开销。因此,遍历TreeMap时强烈推荐使用entrySet方法。

2. 只遍历key时,keySet方法更为合适,因为entrySet将无用的value也给取出来了,浪费了性能和空间。在上述测试结果中,keySet比entrySet方法耗时少24%。

3. 只遍历value时,使用vlaues方法是最佳选择,entrySet也明显优于keySet方法。

4. 在不同的遍历写法中,推荐使用如下写法,其效率略高一些:

for (String key : map.keySet()) {

    value = map.get(key);

}

 

for (Entry<String, String> entry: map.entrySet()) {

    key = entry.getKey();

    value = entry.getValue();

}

 

for (String value : map.values()) {

}

 

How to Iterate Over a Map in Java

java中遍历Map有不少的方法。我们看一下最常用的方法及其优缺点。

既然java中的所有map都实现了Map接口,以下方法适用于任何map实现(HashMap, TreeMap, LinkedHashMap, Hashtable, 等等)

 

方法一 for-each循环中使用entries来遍历

这是最常见的并且在大多数情况下也是最可取的遍历方式。在键值都需要时使用。

1. Map<Integer, Integer> map = new HashMap<Integer, Integer>();  

2.   

3. for (Map.Entry<Integer, Integer> entry : map.entrySet()) {  

4.   

5.     System.out.println("Key = " + entry.getKey() + ", Value = " + entry.getValue());  

6.   

7. }  

注意:for-each循环在java 5中被引入所以该方法只能应用于java 5或更高的版本中。如果你遍历的是一个空的map对象,for-each循环将抛出NullPointerException,因此在遍历前你总是应该检查空引用。

 方法二 for-each循环中遍历keys或values。

如果只需要map中的键或者值,你可以通过keySet或values来实现遍历,而不是用entrySet。

1. Map<Integer, Integer> map = new HashMap<Integer, Integer>();  

2. //遍历map中的键  

3. for (Integer key : map.keySet()) {  

4.   

5.     System.out.println("Key = " + key);  

6. }  

7. //遍历map中的值  

8.   

9. for (Integer value : map.values()) {  

10.   

11.     System.out.println("Value = " + value);  

12.   

13. }  

该方法比entrySet遍历在性能上稍好(快了10%),而且代码更加干净。

方法三使用Iterator遍历

使用泛型:

1. Map<Integer, Integer> map = new HashMap<Integer, Integer>();  

2.   

3. Iterator<Map.Entry<Integer, Integer>> entries = map.entrySet().iterator();  

4.   

5. while (entries.hasNext()) {  

6.   

7.     Map.Entry<Integer, Integer> entry = entries.next();  

8.   

9.     System.out.println("Key = " + entry.getKey() + ", Value = " + entry.getValue());  

10.   

11. }  

不使用泛型:

1. Map map = new HashMap();  

2.   

3. Iterator entries = map.entrySet().iterator();  

4.   

5. while (entries.hasNext()) {  

6.   

7.     Map.Entry entry = (Map.Entry) entries.next();  

8.   

9.     Integer key = (Integer)entry.getKey();  

10.   

11.     Integer value = (Integer)entry.getValue();  

12.   

13.     System.out.println("Key = " + key + ", Value = " + value);  

14.   

15. }  

 

你也可以在keySet和values上应用同样的方法。

该种方式看起来冗余却有其优点所在。首先,在老版本java中这是惟一遍历map的方式。另一个好处是,你可以在遍历时调用iterator.remove()来删除entries,另两个方法则不能。根据javadoc的说明,如果在for-each遍历中尝试使用此方法,结果是不可预测的。

从性能方面看,该方法类同于for-each遍历(即方法二)的性能。

方法四、通过键找值遍历(效率低)

[java] view plain copy

1. Map<Integer, Integer> map = new HashMap<Integer, Integer>();  

2.   

3. for (Integer key : map.keySet()) {  

4.   

5.     Integer value = map.get(key);  

6.   

7.     System.out.println("Key = " + key + ", Value = " + value);  

8.   

9. }  

 

作为方法一的替代,这个代码看上去更加干净;但实际上它相当慢且无效率。因为从键取值是耗时的操作(与方法一相比,在不同的Map实现中该方法慢了20%~200%)。如果你安装了FindBugs,它会做出检查并警告你关于哪些是低效率的遍历。所以尽量避免使用。

 

总结

如果仅需要键(keys)或值(values)使用方法二。如果你使用的语言版本低于java 5,或是打算在遍历时删除entries,必须使用方法三。否则使用方法一(键值都要)。

 

通常来说,Map是一个由键值对组成的数据结构,且在集合中每个键是唯一的。下面就以K和V来代表键和值,来说明一下java中关于Map的九大问题。

0、将Map转换为List类型

java中Map接口提供了三种集合获取方式:Key set,,value set, and key-value set.。它们都可以通过构造方法或者addAll()方法来转换为List类型。下面代码就说明了如何从Map中构造ArrayList:

// key list
List keyList = new ArrayList(map.keySet());
// value list
List valueList = new ArrayList(map.valueSet());
// key-value list
List entryList = new ArrayList(map.entrySet());

1、通过Entry 遍历Map

java中这种以键值对存在的方式被称为Map.Entry。Map.entrySet()返回的是一个key-value 集合,这是一种非常高效的遍历方式。

for(Entry entry: map.entrySet()) {
// get key
K key = entry.getKey();
// get value
V value = entry.getValue();
}

Iterator 我们也经常用到,尤其是在JDK1.5以前

Iterator itr = map.entrySet().iterator();
while(itr.hasNext()) {
Entry entry = itr.next();
// get key
K key = entry.getKey();
// get value
V value = entry.getValue();
}

2、通过Key来对Map排序

排序需要对Map的ke进行频繁的操作,一种方式就是通过比较器(comparator )来实现:

List list = new ArrayList(map.entrySet());
Collections.sort(list, new Comparator() {

@Override
public int compare(Entry e1, Entry e2) {
return e1.getKey().compareTo(e2.getKey());
}

});

另外一种方法就是通过SortedMap,但必须要实现Comparable接口。

SortedMap sortedMap = new TreeMap(new Comparator() {

@Override
public int compare(K k1, K k2) {
return k1.compareTo(k2);
}

});
sortedMap.putAll(map);

3、对value对Map进行排序

这与上一点有些类似,代码如下:

List list = new ArrayList(map.entrySet());
Collections.sort(list, new Comparator() {

@Override
public int compare(Entry e1, Entry e2) {
return e1.getValue().compareTo(e2.getValue());
}

});

4、初始化一个static 的常量Map

当你希望创建一个全局静态Map的时候,我们有以下两种方式,而且是线程安全的。
而在Test1中,我们虽然声明了map是静态的,但是在初始化时,我们依然可以改变它的值,就像Test1.map.put(3,”three”);
Test2中,我们通过一个内部类,将其设置为不可修改,那么当我们运行Test2.map.put(3,”three”)的时候,它就会抛出一个UnsupportedOperationException 异常来禁止你修改。

public class Test1 {

private static final Map map;
static {
map = new HashMap();
map.put(1, “one”);
map.put(2, “two”);
}
}

public class Test2 {

private static final Map map;
static {
Map aMap = new HashMap();
aMap.put(1, “one”);
aMap.put(2, “two”);
map = Collections.unmodifiableMap(aMap);
}
}

5、HashMap, TreeMap, and Hashtable之间的不同

Map接口中,共有三种实现:HashMap,TreeMap,Hashtable。

它们之间各有不同,详细内容请参考《 HashMap vs. TreeMap vs. Hashtable vs. LinkedHashMap》一文。

6、Map中的反向查询

我们在Map添加一个键值对后,意味着这在Map中键和值是一一对应的,一个键就是对应一个值。但是有时候我们需要反向查询,比如通过某一个值来查找它的键,这种数据结构被称为bidirectional map,遗憾的是JDK并没有对其支持。

Apache和Guava 共同提供了这种bidirectional map实现,它在实现中它规定了键和值都是必须是1:1的关系。

7、对Map的复制

java中提供了很多方法都可以实现对一个Map的复制,但是那些方法不见得会时时同步。简单说,就是一个Map发生的变化,而复制的那个依然保持原样。下面是一个比较高效的实现方法:

Map copiedMap = Collections.synchronizedMap(map);

当然还有另外一个方法,那就是克隆。但是我们的java鼻祖Josh Bloch却不推荐这种方式,他曾经在一次访谈中说过关于Map克隆的问题:在很多类中都提供了克隆的方法,因为人们确实需要。但是克隆非常有局限性,而且在很多时候造成了不必要的影响。(原文《Copy constructor versus cloning》)

8、创建一个空的Map

如果这个map被置为不可用,可以通过以下实现

map = Collections.emptyMap();

相反,我们会用到的时候,就可以直接

map = new HashMap();

 

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值