Map集合中value()、keySet()和entrySet()以及性能的分析

来源:互联网 发布:日本手办产业数据 编辑:程序博客网 时间:2024/06/03 23:37

在Map集合中

values():方法是获取集合中的所有的值----没有键,没有对应关系,

KeySet():将Map中所有的键存入到set集合中。因为set具备迭代器。所有可以迭代方式取出所有的键,再根据get方法。获取每一个键对应的值。 keySet():迭代后只能通过get()取key 
entrySet():Set<Map.Entry<K,V>> entrySet() //返回此映射中包含的映射关系的 Set 视图。 Map.Entry表示映射关系。entrySet():迭代后可以e.getKey(),e.getValue()取key和value。返回的是Entry接口 。

下面通过实例来看看:

[html] view plain copy
  1. public class map {  
  2.     public static void main(String[] args) {  
  3.         Map<String, String> maps = new HashMap<String,String>();  
  4.         maps.put("10","AA");  
  5.         maps.put("11","BB");  
  6.         maps.put("12","CC");  
  7.         maps.put("13","DD");  
  8.         maps.put("14","EE");  
  9.         //①value():方法是获取集合中的所有的值----没有键,没有对应关系  
  10.          Collection<String> collections =  maps.values();  
  11.          System.out.println(collections);//[AA, DD, EE, BB, CC]  
  12.         //②keySet():获取集合中所有的键  
  13.          Set<String> sets = maps.keySet();  
  14.          System.out.println(sets);//[10, 13, 14, 11, 12]  
  15.          Iterator<String> iterators = sets.iterator();  
  16.          while(iterators.hasNext()){  
  17.              String key = iterators.next();  
  18.              System.out.print(" "+key);// 10 13 14 11 12  
  19.              String val = maps.get(key);  
  20.              System.out.print(" "+val);//AA DD EE BB CC  
  21.          }  
  22.          //③entrySet():返回此映射中包含的映射关系的 Set 视图。 Map.Entry表示映射关系  
  23.          //通过entrySet()方法将map集合中的映射关系取出(这个关系就是Map.Entry类型)  
  24.          Set<Entry<String, String>> entries = maps.entrySet();  
  25.          //将关系集合entrySet进行迭代,存放到迭代器中   
  26.          Iterator<Entry<String, String>> iterator1 =  entries.iterator();  
  27.          while(iterator1.hasNext()){  
  28.              Entry<String, String> entry = iterator1.next();//获取Map.Entry关系对象entry  
  29.              String key1 = entry.getKey();  
  30.              String value1 = entry.getValue();  
  31.              System.out.println(key1+"--->"+value1);  
  32.          }   
  33.     }  
  34. }  
说明:

①Set<K> keySet():返回值是个只存放key值的Set集合(集合中无序存放的)迭代后只能通过get()取key
②Set<Map.Entry<K,V>> entrySet():返回映射所包含的映射关系的Set集合(一个关系就是一个键-值对),就是把(key-value)作为一个整体一对一对地存放到Set集合当中的。迭代后可以e.getKey(),e.getValue()取key和value。返回的是Entry接口

虽然使用keyset及entryset来进行遍历能取得相同的结果但两者的遍历速度是有差别的,keySet()的速度比entrySet()慢了很多也就是keySet方式遍历Map的性能不如entrySet性能好为了提高性能,以后多考虑用entrySet()方式来进行遍历。

问题:为什么keySet方式遍历Map的性能不如entrySet性能

[html] view plain copy
  1. public class map1 {  
  2.     public static void main(String[] args) {  
  3.     <span style="font-family:Tahoma, ";;;;">        </span>//为什么keySet性能不如entrySet呢?实验证明  
  4.              HashMap<String, String> keySetMap = new HashMap<String, String>();     
  5.          HashMap<String, String> entrySetMap = new HashMap<String, String>();   
  6.          for (int i = 0; i < 1000000; i++) {     
  7.              keySetMap.put("" + i, "keySet");     
  8.          }     
  9.          for (int j = 0; j < 1000000; j++) {     
  10.              entrySetMap.put("" + j, "entrySet");     
  11.          }   
  12.          //keySet实验:1000000条数据,用时54  
  13.          long startTimeOne = System.currentTimeMillis();     
  14.          Iterator<String> keySetIterator = keySetMap.keySet().iterator();     
  15.          while (keySetIterator.hasNext()) {     
  16.              String key = keySetIterator.next();     
  17.              String value = keySetMap.get(key);<span style="color:#FF6666;"><strong>//性能差距的原因:取得key所对应的value时,此时还要访问Map的这个方法</strong></span>     
  18.              System.out.println(value);     
  19.          }     
  20.          System.out.println("keyset spent times:"    
  21.                  + (System.currentTimeMillis() - startTimeOne));//54     
  22.          //entrySet实验:1000000条数据,用时28  
  23.          long startTimeTwo = System.currentTimeMillis();     
  24.          Iterator<Entry<String, String>> entryKeyIterator = entrySetMap     
  25.                  .entrySet().iterator();     
  26.          while (entryKeyIterator.hasNext()) {     
  27.              Entry<String, String> e = entryKeyIterator.next();     
  28.              System.out.println(e.getValue());     
  29.          }     
  30.          System.out.println("entrySet spent times:"    
  31.                  + (System.currentTimeMillis() - startTimeTwo));//28   
  32.     }  
  33.  }   
原因分析:

通过查看源代码发现,调用这个方法keySetMap.keySet()会生成KeyIterator迭代器,其next方法只返回其key值. 
Java代码  

[html] view plain copy
  1. private class KeyIterator extends HashIterator<K> {     
  2.        public K next() {     
  3.            return nextEntry().getKey();     
  4.        }     
  5.    }  
而调用entrySetMap.entrySet()方法会生成EntryIterator 迭代器,其next方法返回一个Entry对象的一个实例,其中包含key和value. 
Java代码  
[html] view plain copy
  1. private class EntryIterator extends HashIterator<Map.Entry<K,V>> {     
  2.        public Map.Entry<K,V> next() {     
  3.            return nextEntry();     
  4.        }     
  5. }   
二者在此时的性能应该是相同的,但方式一再取得key所对应的value时,此时还要访问Map的这个方法,这时,方式一多遍历了一次table.
也就是上述案例中的
[html] view plain copy
  1. String value = keySetMap.get(key)  

查看源码可以看出



这个方法就是二者性能差别的主要原因. 

阅读全文
0 0
原创粉丝点击