Map接口:
Map用于保存具有映射关系的数据,因此Map集合里都保存着两组值,一组值用于保存Map里面的key,另外一组用于保存Map里的value,key和value都可以是任何引用类型的数据。Map中的key不允许重复,即同一个Map对象的任何两个key通过equals方法比较总是返回false。
key和value之间存在单向的一对一的关系,即通过指定的key值,能够唯一的找到确定的value值。Map的get方法就是给出指定的key,来取出其对应的value。
值得注意的是:
1.如果把Map里的所有key放在一起来看,它们就组成了一个set集合(所有的key没有顺序,key与key之间不能重复),实际上Map包含了一个keyset()的方法,用于返回Map里的所有key组成的set集合。
2.如果把Map里的所有value放在一起来看,它们又非常类似于一个List:元素与元素之间可以重复,每个元素都可以根据索引值来进行查找,只是Map中的索引不再使用下标的整数值,而是以另一个对象(key)来作为索引。如果需要从List集合中取出元素,需要提供该元素的数字索引;如果需要从Map中取出元素,则需要提供该元素的key索引,下面通过例1来说明一下Map中的一些方法:
下面的例一表示了Map类的基本功能:
例一:
import java.util.HashMap;
import java.util.Map;
public class MapTest2 {
public static void main(String[] args)
{
Map map = new HashMap();
// 成对放入多个key-value对
map.put("人森苦短" , 109);
map.put("疯狂java", 10);
map.put("ee轰炸机" , 79);
// 多次放入的key-value对中value可以重复
map.put("疯狂java",99);
// 放入重复的key时,新的value会覆盖原有的value
// 如果新的value覆盖了原有的value,该方法返回被覆盖的value
System.out.println(map.put("ee轰炸机",99)); // 输出10
System.out.println(map); // 输出的Map集合包含4个key-value对
// 判断是否包含指定key
System.out.println("是否包含值为ee轰炸机key:"
+ map.containsKey("ee轰炸机")); // 输出true
// 判断是否包含指定value
System.out.println("是否包含值为 99 value:"
+ map.containsValue(99)); // 输出true
// 获取Map集合的所有key组成的集合,通过遍历key来实现遍历所有key-value对
for (Object key : map.keySet() )
{
// map.get(key)方法获取指定key对应的value
System.out.println(key + "-->" + map.get(key));
}
map.remove("疯狂java"); // 根据key来删除key-value对,如果是两个key相同而value不同的键值对,则都删除
System.out.println(map); // 输出结果不再包含疯狂java的key-value对
}
}
HashMap类:
HashMap作为Map接口的一个实现类,有如下三种特点:
一.HashMap是Map的一个重要实现类,也是最常用的,基于哈希表实现;
二.HashMap中的Entry对象是无序排列的;
三.key值和value值都可以为null,但是一个HashMap只能有一个key值为null的映射(key值不可重复)
下面再通过一个案例二来加深对HashMap的理解:
案例说明:
通过Map<String,Student>进行学生信息管理,其中key为学生id,value为学生对象;
通过键盘输入学生信息;
对集合中的学生信息进行增删改查操作;
例二:
student类:
import java.util.HashSet;
import java.util.Set;
/**
* 学生类
*/
public class Student {
private String id;
private Set<Course> courses;
public Set<Course> getCourses() {
return courses;
}
public void setCourses(Set<Course> courses) {
this.courses = courses;
}
private String name;
public String getId() {
return id;
}
public void setId(String id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public Student(String id, String name) {
this.id = id;
this.name = name;
this.courses = new HashSet<Course>();
}
}
MapTest类:
package Map;
import java.util.HashMap;
import java.util.Map;
import java.util.Map.Entry;
import java.util.Scanner;
import java.util.Set;
public class MapTest {
public Map<String, Student> students;//创建一个Map类型的属性,用来装学生类型对象
//规定其key和value值
/**
* 在构造器中初始化students属性
*/
public MapTest() {
this.students = new HashMap<String, Student>();
}
/**
* 测试添加:输入学生ID,判断是否被占用 若未被占用,则输入姓名,创建新学生对象,并且添加到students中
*/
public void testPut() {
// 创建一个Scanner对象,用来获取输入的学生ID和姓名
Scanner console = new Scanner(System.in);
int i = 0;
while (i < 3) {
System.out.println("请输入学生ID:");
String ID = console.next();
// 判断该ID是否被占用
//Map的get方法是根据键值来取出其对应的value值,若不存在则返回null
Student st = students.get(ID);
if (st == null) {
// 提示输入学生姓名
System.out.println("请输入学生姓名:");
String name = console.next();
// 创建新的学生对象
Student newStudent = new Student(ID, name);
// 通过调用students的put方法,添加ID-学生映射
students.put(ID, newStudent);
System.out.println("成功添加学生:" + students.get(ID).getName());
i++;
} else {
System.out.println("该学生ID已被占用!");
continue;
}
}
}
/**
* 测试Map的keySet方法
* 注意打印输出的顺序和添加的顺序是不同的,
* 原因是因为这里用了hashmap来实现接口,
* 而hashmap中的映射是无序的
*/
public void testKeySet() {
// 通过keySet方法,返回Map中的所有“键”的Set集合
Set<String> keySet = students.keySet();
// 取得students的容量
//通过size方法获得map的容量
System.out.println("总共有:" + students.size() + "个学生!");
// 遍历keySet,取得每一个键,再调用get方法取得每个键对应的value
for (String stuId : keySet) {
Student st = students.get(stuId);
if (st != null)
System.out.println("学生:" + st.getName());
}
}
/**
* 测试删除Map中的映射
*/
public void testRemove() {
// 获取从键盘输入的待删除学生ID字符串
Scanner console = new Scanner(System.in);
while (true) {
// 提示输入待删除的学生的ID
System.out.println("请输入要删除的学生ID!");
String ID = console.next();
// 判断该ID是否有对应的学生对象
Student st = students.get(ID);
if (st == null) {
// 提示输入的ID并不存在
System.out.println("该ID不存在!");
continue;
}
students.remove(ID);
System.out.println("成功删除学生:" + st.getName());
break;
}
}
/**
* 通过entrySet方法来遍历Map
*/
public void testEntrySet() {
// 通过entrySet方法,返回Map中的所有键值对
//注意entry本身是带有泛型的
Set<Entry<String, Student>> entrySet = students.entrySet();
//entrySet返回的是所有键值对的集合
for (Entry<String, Student> entry : entrySet) {
System.out.println("取得键:" + entry.getKey());
System.out.println("对应的值为:" + entry.getValue().getName());
}
}
/**
* 利用put方法修改Map中的已有映射
*/
public void testModify() {
// 提示输入要修改的学生ID
System.out.println("请输入要修改的学生ID:");
// 创建一个Scanner对象,去获取从键盘上输入的学生ID字符串
Scanner console = new Scanner(System.in);
while (true) {
// 取得从键盘输入的学生ID
String stuID = console.next();
// 从students中查找该学生ID对应的学生对象
Student student = students.get(stuID);
if (student == null) {
System.out.println("该ID不存在!请重新输入!");
continue;
}
// 提示当前对应的学生对象的姓名
System.out.println("当前该学生ID,所对应的学生为:" + student.getName());
// 提示输入新的学生姓名,来修改已有的映射
System.out.println("请输入新的学生姓名:");
String name = console.next();
Student newStudent = new Student(stuID, name);
students.put(stuID, newStudent);
System.out.println("修改成功!");
break;
}
}
/**
* @param args
*/
public static void main(String[] args) {
MapTest mt = new MapTest(); //创建一个MapTest对象的实例
mt.testPut();
mt.testKeySet();
// mt.testRemove();
// mt.testEntrySet();
// mt.testModify();
// mt.testEntrySet();
}
}