how2j.cn

本视频是解读性视频,所以希望您已经看过了本知识点的内容,并且编写了相应的代码之后,带着疑问来观看,这样收获才多。 不建议一开始就观看视频



3分12秒
本视频采用html5方式播放,如无法正常播放,请将浏览器升级至最新版本,推荐火狐,chrome,360浏览器 如果装有迅雷,播放视频呈现直接下载状态,请调整 迅雷系统设置-基本设置-启动-监视全部浏览器 (去掉这个选项)



示例 1 : HashMap的键值对   
示例 2 : 键不能重复,值可以重复   
示例 3 : 练习-查找内容性能比较   
示例 4 : 答案-查找内容性能比较   

示例 1 :

HashMap的键值对

HashMap储存数据的方式是—— 键值对
package collection; import java.util.HashMap; public class TestCollection { public static void main(String[] args) { HashMap<String,String> dictionary = new HashMap<>(); dictionary.put("adc", "物理英雄"); dictionary.put("apc", "魔法英雄"); dictionary.put("t", "坦克"); System.out.println(dictionary.get("t")); } }
package collection;
  
import java.util.HashMap;
  
public class TestCollection {
    public static void main(String[] args) {
        HashMap<String,String> dictionary = new HashMap<>();
        dictionary.put("adc", "物理英雄");
        dictionary.put("apc", "魔法英雄");
        dictionary.put("t", "坦克");
        
        System.out.println(dictionary.get("t"));
    }
}
示例 2 :

键不能重复,值可以重复

对于HashMap而言,key是唯一的,不可以重复的。
所以,以相同的key 把不同的value插入到 Map中会导致旧元素被覆盖,只留下最后插入的元素。
不过,同一个对象可以作为值插入到map中,只要对应的key不一样
package collection; import java.util.HashMap; import charactor.Hero; public class TestCollection { public static void main(String[] args) { HashMap<String,Hero> heroMap = new HashMap<String,Hero>(); heroMap.put("gareen", new Hero("gareen1")); System.out.println(heroMap); //key为gareen已经有value了,再以gareen作为key放入数据,会导致原英雄,被覆盖 //不会增加新的元素到Map中 heroMap.put("gareen", new Hero("gareen2")); System.out.println(heroMap); //清空map heroMap.clear(); Hero gareen = new Hero("gareen"); //同一个对象可以作为值插入到map中,只要对应的key不一样 heroMap.put("hero1", gareen); heroMap.put("hero2", gareen); System.out.println(heroMap); } }
示例 3 :

练习-查找内容性能比较

Or  姿势不对,事倍功半! 点击查看做练习的正确姿势
准备一个ArrayList其中存放3000000(三百万个)Hero对象,其名称是随机的,格式是hero-[4位随机数]
hero-3229
hero-6232
hero-9365
...

因为总数很大,所以几乎每种都有重复,把名字叫做 hero-5555的所有对象找出来
要求使用两种办法来寻找
1. 不使用HashMap,直接使用for循环找出来,并统计花费的时间
2. 借助HashMap,找出结果,并统计花费的时间
示例 4 :

答案-查找内容性能比较

在查看答案前,尽量先自己完成,碰到问题再来查看答案,收获会更多
本视频是解读性视频,所以希望您已经看过了本答案的内容,带着疑问来观看,这样收获才多。 不建议一开始就观看视频

6分3秒 本视频采用html5方式播放,如无法正常播放,请将浏览器升级至最新版本,推荐火狐,chrome,360浏览器 如果装有迅雷,播放视频呈现直接下载状态,请调整 迅雷系统设置-基本设置-启动-监视全部浏览器 (去掉这个选项)


需要注意的是:
Map的key是字符串,英雄的名称
Map的value是List,里面放了名称相同的多个英雄

如图所示,使用Map查找会快非常多,为什么会快很多呢? 这就涉及到hashcode 原理,将在后续章节专门详细讲解。
答案-查找内容性能比较
package collection; import java.util.ArrayList; import java.util.HashMap; import java.util.List; import charactor.Hero; public class TestCollection { public static void main(String[] args) { List<Hero> hs =new ArrayList<>(); System.out.println("初始化开始"); for (int i = 0; i < 3000000; i++) { Hero h = new Hero( "hero-" + random()); hs.add(h); } //名字作为key //名字相同的hero,放在一个List中,作为value HashMap<String,List<Hero>> heroMap =new HashMap(); for (Hero h : hs) { List<Hero> list= heroMap.get( h.name); if(list==null){ list = new ArrayList<>(); heroMap.put(h.name, list); } list.add(h); } System.out.println("初始化结束"); System.out.println("开始查找"); findByIteration(hs); findByMap(heroMap); } private static List<Hero> findByMap(HashMap<String,List<Hero>> m) { long start =System.currentTimeMillis(); List <Hero>result= m.get("hero-5555"); long end =System.currentTimeMillis(); System.out.printf("通过map查找,一共找到%d个英雄,耗时%d 毫秒%n",result.size(),end-start); return result; } private static List<Hero> findByIteration (List<Hero> hs) { long start =System.currentTimeMillis(); List<Hero> result =new ArrayList<>(); for (Hero h : hs) { if(h.name.equals("hero-5555")){ result.add(h); } } long end =System.currentTimeMillis(); System.out.printf("通过for查找,一共找到%d个英雄,耗时%d 毫秒%n", result.size(),end-start); return result; } public static int random(){ return ((int)(Math.random()*9000)+1000); } }


HOW2J公众号,关注后实时获知布最新的教程和优惠活动,谢谢。


问答区域    
2018-10-07 运行list溢出
Bug6
要是ArrayList存的是int、float的基本数据类型,其最大的容量是2的21次方,要是存的是纯对象类型,其最大容量是2的19次方,它的限制条件的很大一个是内存问题,就是计算机的堆空间用完了。 最后的练习题,我写的代码发生内存溢出,然后查资料之后把300万改成30万,然后程序一切正常。站长,您运行为什么没有出现溢出现象?








答案 或者 代码至少填写一项, 如果是自己有问题,请重新提问,否则站长有可能看不到




2018-09-24 站长的理论知识很少,要仔细看下这个代码,有点绕,不太好理解,注释多写了点
Hello_Cyoung



for遍历查找方法找到的名字为“hero-5555”的对象个数为:323 for遍历查找方法耗时:114 HashMap查找方法找到的名字为“hero-5555”的对象个数为:323 HashMap查找方法耗时:0
/**
 * 
 */
package setFrame;

import java.util.ArrayList;
import java.util.HashMap;
import java.util.List;

/**
 * @author Administrator
 *
 */
public class HashMapSet {
	public static void main(String[] args) {
		/*
		 * 准备一个ArrayList其中存放3000000(三百万个)Hero对象,其名称是随机的,格式是hero-[4位随机数]
		 * hero-3229
		 * hero-6232
		 * hero-9365
		 * ...
		 * 因为总数很大,所以几乎每种都有重复,把名字叫做 hero-5555的所有对象找出来
		 * 要求使用两种办法来寻找
		 * 1. 不使用HashMap,直接使用for循环找出来,并统计花费的时间
		 * 2. 借助HashMap,找出结果,并统计花费的时间
		 */
		
		
		//创建3000000个随机名字的hero
		List<Hero> heroList = new ArrayList<>();
		for(int i=0;i<3000000;i++) {
			Hero h = new Hero("hero-"+(int)(Math.random()*9000+1000));
			heroList.add(h);
		}
//		for(Hero each : heroList)
//			System.out.println(each);
		
		//hero的name作为key,名字相同的hero作为Value存入HashMap中(奇思妙想):
		HashMap<String,List<Hero>> heroHashMap = new HashMap<>();//新建一个HashMap用来存储heroList中的值
		for(Hero each : heroList) { //遍历heroList
				List<Hero> list = heroHashMap.get(each.name);//将heroHashMap中key为each。name的list引用给list
				if(list == null) {//如果list为空(当一个新的name进来时list肯定为空,因为heroHashMap中没有这个新name的key)
					list = new ArrayList();//新建一个ArrayList对象,将其引用为list的指向
					heroHashMap.put(each.name,list);//将这个each。name作为key,新建的ArrayList作为value存到heroHashMap中
				}
				list.add(each);//如果是出现过的name,就不需要新建ArrayList了,直接加到已有的list中
			}
		
		//i作为key,hero的name作为value存入HashMap中(不适用)
//		HashMap<Integer,String> heroHashMap = new HashMap<>();
//		for(int i=0;i<heroList.size();i++) {
//			heroHashMap.put(i,heroList.get(i).name);
////			System.out.println(heroHashMap.get(i));
//		}
		forSearch(heroList);
		hashMapSearch(heroHashMap);
	}
	
	//第一种方法:不使用HashMap,直接使用for循环找出来,并统计花费的时间
	public static void forSearch(List<Hero> list) {
		long start = System.currentTimeMillis();
		int num = 0;
		for(int i=0;i<list.size();i++) {
			if(list.get(i).name.equals("hero-5555"))
				num++;
		}
		System.out.println("for遍历查找方法找到的名字为“hero-5555”的对象个数为:"+num);
		long end = System.currentTimeMillis();
		System.out.println("for遍历查找方法耗时:"+(end-start));
	}
	
	//第二种方法:借助HashMap,找出结果,并统计花费的时间
	public static void hashMapSearch(HashMap<String,List<Hero>> heroHashMap) {
		long start = System.currentTimeMillis();
//		int num = 0;
//		HashMap<Integer,String> heroHashMap = new HashMap<>();
//		for(int i=0;i<list.size();i++) {
//			heroHashMap.put(i,list.get(i).name);
//		}
		List<Hero> result = heroHashMap.get("hero-5555");//将heroHashMap中的key为“hero-5555”的List赋给result这个List!
		System.out.println("HashMap查找方法找到的名字为“hero-5555”的对象个数为:"+result.size());
		long end = System.currentTimeMillis();
		System.out.println("HashMap查找方法耗时:"+(end-start));
	}
}


/*
//第一种:普遍使用,二次取值
 System.out.println("通过Map.keySet遍历key和value:");
 for (String key : map.keySet()) {
  System.out.println("key= "+ key + " and value= " + map.get(key));
 }
  
 //第二种
 System.out.println("通过Map.entrySet使用iterator遍历key和value:");
 Iterator<Map.Entry<String, String>> it = map.entrySet().iterator();
 while (it.hasNext()) {
  Map.Entry<String, String> entry = it.next();
  System.out.println("key= " + entry.getKey() + " and value= " + entry.getValue());
 }
  
 //第三种:推荐,尤其是容量大时
 System.out.println("通过Map.entrySet遍历key和value");
 for (Map.Entry<String, String> entry : map.entrySet()) {
  System.out.println("key= " + entry.getKey() + " and value= " + entry.getValue());
 }
 
 //第四种
 System.out.println("通过Map.values()遍历所有的value,但不能遍历key");
 for (String v : map.values()) {
  System.out.println("value= " + v);
 }
}
*/

							


4 个答案

清水浩 答案时间:2018-10-05
已解决!

清水浩 答案时间:2018-10-05
//将heroHashMap中key为each。name的list引用给list 这句话是什么意思啊!!???

yuhlien 答案时间:2018-09-27
创建3000000个随机Hero,耗时: 00:00'890" for循环找到311个hero-5555,耗时: 00:00'104" HashMap找到311个hero-5555,搜索耗时: 00:00'000" HashMap构建+搜索总耗时: 00:00'667" 加上创建hashmap的时间确实要多上不少,但是多次搜索时间就能省下来了。

花木沐 答案时间:2018-09-25
我觉得你的代码有问题 构建HashMap<String,List<Hero>>这个步骤应该放在hashMapSearch方法中,而不是主方法,因为要想用HashMap搜索肯定要先创建,而这个步骤花的时间也不少 这样的话HashMap查找用的时间要比for长




答案 或者 代码至少填写一项, 如果是自己有问题,请重新提问,否则站长有可能看不到





2018-09-05 内容性能比较问题
2018-08-17 查找内容性能比较
2018-07-31 交作业
2018-07-11 为啥我照站长的答案改的,hashmap的性能和arraylist是一样的啊?
2018-07-10 我比较笨,站长的代码我理解不了,自己找的方法做的
2018-05-07 方法体差不多 性能不一样
2018-05-04 【交作业】 查找内容性能比较(存在疑问)
2018-04-17 站长答案中一段代码不理解,渴求懂的人解答下?
2018-04-16 两者的比较有问题
2018-03-12 这章节的题目跟上面一样,都是比较性能。
2018-02-08 有段代码没看懂,求大神解释
2018-01-20 分享一下关于hashmap遍历的几种方式
2017-09-12 空指针异常!!!!!!
2017-08-16 效率问题提问
2017-07-29 有一段代码看着不是很清楚
2017-03-29 练习题:查找内容性能比较有点没懂
2016-10-26 练习题的问题




提问之前请登陆
关于 JAVA 中级-集合框架-HashMap 的提问

尽量提供截图代码异常信息,有助于分析和解决问题。 也可进本站QQ群交流: 902680467
提问尽量提供完整的代码,环境描述,越是有利于问题的重现,您的问题越能更快得到解答。
对教程中代码有疑问,请提供是哪个步骤,哪一行有疑问,这样便于快速定位问题,提高问题得到解答的速度
在已经存在的几千个提问里,有相当大的比例,是因为使用了和站长不同版本的开发环境导致的,比如 jdk, eclpise, idea, mysql,tomcat 等等软件的版本不一致。
请使用和站长一样的版本,可以节约自己大量的学习时间。 站长把教学中用的软件版本整理了,都统一放在了这里, 方便大家下载: http://how2j.cn/k/helloworld/helloworld-version/1718.html

上传截图