Java集合之Map接口与实现类详解

 更新时间:2022年12月08日 09:53:59   作者:从未止步..  
这篇文章主要为大家详细介绍了Java集合中的Map接口与实现类,文中的示例代码讲解详细,对我们学习Java有一定的帮助,感兴趣的可以了解一下

初识Map

Map接口没有从Collection接口继承,Map接口用于维护“键-值”对数据,这个“键-值”对就是Map中的元素,Map提供“键(Key)”到“值(value)”的映射,一个Map中键值必须是唯一的,不能有重复的键,因为Map中的“键-值”对元素是通过键来唯一标识的,Map的键是用Set集合来存储的,所以充当键元素的类必须重写hashCode()和equals()方法,通过键元素来检索对应值元素

Map中常用方法

V put(K key, V value)	//向Map集合中添加键值对 

V get(Object key)			//返回指定键映射到的值,通过key获取value 

void clear()		//从此映射中删除所有映射,清空Map集合

boolean containsKey(Object key)	//判断Map是否包含某个key

boolean containsValue(Object value)	//判断Map是否包含某个value

boolean isEmpty()	//判断Map中元素个数是否为0

Set<K> keySet()	//获取Map集合中所有的key(所有的键都是一个“Set集合”)

V remove(Object key)	//通过key删除键值对,如果存在,则从该映射中移除键的映射(可选操作)

int size()	//获取Map集合中键值对的个数

HashMap

HashMap特点

(1)HashMap实现了Map 接口

(4)HashMap以Hash表为基础构建

(2)HashMap 允许键(Key)为空和值(Value)为空

(3)HashMap不是线程安全的,不能用在多线程环境中

HashMap常用的构造方法如下

(1)HashMap():创建一个空 HashMap 对象,并为其分配默认的初始容量和加载因子

(2)HashMap(int initialCapacity):创建一个空 HashMap 对象,并为其分配指定的初始容量

(3)HashMap(int initialCapacity,float loadFactor);创建一个空 HashMap 对象,并其分配指定的初始容量和加载因子

(4)HashMap(Map m):创建一个与给定Map 对象具有相同键-值对的 HashMap 对象

HashMap方法的简单使用

import java.security.Key;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Iterator;
import java.util.Set;
public class Warehouse1 {
    public static void main(String[] args){
        HashMap<String,Integer> hashMap=new HashMap<>();
        hashMap.put("年龄",20);
        hashMap.put("身高",180);
        hashMap.put("体重",60);
        System.out.println("键为年龄所对应的值为:"+hashMap.get("年龄"));
        System.out.println("该map是否包含键为“年龄”?"+hashMap.containsKey("年龄"));
        System.out.println("该map是否包含键值为90?"+hashMap.containsValue(90));
        System.out.println("该map是否为空?"+hashMap.isEmpty());
        System.out.println("该map的长度大小为:"+hashMap.size());
        System.out.println("移除键为“体重“的键值对"+hashMap.remove("体重"));
        System.out.println("哈希映射中的内容如下:"+hashMap);
        Set<String> keys=hashMap.keySet();
        Iterator<String> iterator=keys.iterator();
        System.out.println("hashmap中的元素如下:");
        while(iterator.hasNext()){
            String s= iterator.next();
            Integer name= hashMap.get(s);
            System.out.println(s+":"+name);
        }
    }
}

输出:

键为年龄所对应的值为:20
该map是否包含键为“年龄”?true
该map是否包含键值为90?false
该map是否为空?false
该map的长度大小为:3
移除键为“体重“的键值对60
哈希映射中的内容如下:{年龄=20, 身高=180}
hashmap中的元素如下:
年龄:20
身高:180

小tips:在输出map中的元素时,我们不能直接使用iterator[迭代器],对元素进行遍历输出,因为迭代只能用在Set/List这种单列存放数据的集合中,而map是双列的,并且也不能使用foreach语句对map进行输出,因为foreach语句的本质也是迭代,只能用在单列集合中。

LinkedHashMap

LinkedHashMap有如下特点

1:LinkedHashMap实现了Map接口

2:LinkedHashMap继承了HashMap,内部还有一个双向链表维护键-值对的顺序,每个键-值对即位于哈希表中,也位于双向链表中

3:LinkedHashMap保存了两种元素顺序,输入顺序和访问顺序,插入顺序是指先添加的元素在前面,后添加的元素在后面,修改对应的元素不会影响元素的顺序,访问顺序指的是get/put操作,对一个键执行put/get操作后,其对应的键-值对会移动到链表末尾,所以最末尾的元素是最近访问的,最开始的是最久没有被访问的

4:LinkedHashMap不是线程安全的,不能用在多线程环境中

LinkedHashMap的构造方法如下

1:LinkedHashMap();构造一个空的LinkedHashMap对象

2:LinkedHashMap(Map m);构造一个具有和给定Map相同键-值对的LinkedHashMap对象

3:LinkedHashMap(int capacity):构造一个给定初始容量capacity的LinkedHashMap对象

4:LinkedHashMap(int capacity,float fillRatio):构造一个给定初始容量capacity和填充比fillRatio的LinkedHashMap对象

5:LinkedHashMap(int capacity,float fillRatio,boolean accessOrder):构造一个给定初始容量capacity,填充比fillRatio以及是否被访问顺序的LinkedHashMap对象,access-sOrder为true表示按访问顺序,否则按插入顺序。

注:前四种构造方法创建的LinkedHashMap对象都是按插入顺序

注:fillRatio是装载因子,范围在0-1.0之间,默认是0.75。当实际元素个数/容量 > fillRatio, HashMap自动进行扩容,以保证检索速度

举例:

分别创建两个LinkedHashMap对象-----linkedHashMap和linkedHashMap1

linkedHashMap是以上述五种方法中的第一种构造的LinkedHashMap对象,而linkedHashMap1是以上述五种方法中的第5种构造的LinkedHashMap对象.

代码如下:

import java.security.Key;
import java.util.*;

public class Warehouse1 {
    public static void main(String[] args){
        LinkedHashMap<String,Integer> linkedHashMap=new LinkedHashMap<>();
        linkedHashMap.put("number1",10);
        linkedHashMap.put("number2",20);
        linkedHashMap.put("number3",30);
        linkedHashMap.put("number4",40);
        linkedHashMap.get("number1");
        linkedHashMap.get("number4");
        Set <Map.Entry<String,Integer>> sets=linkedHashMap.entrySet();//Map.Entry是Map的一个内部类,表达一个 key/value映射关系
        System.out.println("linkedHashMap中的元素为:");
        for(Map.Entry<String,Integer> entry:sets) {
            System.out.println("键为:" + entry.getKey() + ",值为:" + entry.getValue());
        }
           System.out.println("---------------");
           LinkedHashMap<String,Integer> linkedHashMap1=new LinkedHashMap<String,Integer>(3,0.5f,true);
            linkedHashMap1.put("number1",10);
            linkedHashMap1.put("number2",20);
            linkedHashMap1.put("number3",30);
            linkedHashMap1.put("number4",40);
            linkedHashMap1.get("number1");
            linkedHashMap1.get("number4");
            linkedHashMap1.put("number3",1000);
            Set<Map.Entry<String,Integer>> set=linkedHashMap1.entrySet();
            System.out.println("linkedHashMap1中的元素为:");
            for(Map.Entry<String ,Integer> entry1:set) {
                System.out.println("键为:" + entry1.getKey() + "值为:" + entry1.getValue());
            }
    }
}

输出:

linkedHashMap中的元素为:
键为:number1,值为:10
键为:number2,值为:20
键为:number3,值为:30
键为:number4,值为:40
---------------
linkedHashMap1中的元素为:
键为:number2值为:20
键为:number1值为:10
键为:number4值为:40
键为:number3值为:1000

通过输出结果,我们不难发现,linkedHashMap中的元素输出的顺序为我们插入时的顺序,而linkedHashMap1中的元素输出顺序为访问顺序,由于这种构造对象的方法,我们设置了access-sOrder为true,因此输出顺序按访问顺序,linkedHashMap1.get(“number1”),表示访问number1,访问完之后,该键值对就移动到最后,下面两行的操作产生的影响也是这样linkedHashMap1.get(“number4”);linkedHashMap1.put(“number3”,1000)。

当按照访问顺序时,put和get去操作已经存在的Entry时,都会把Entry移动到双向链表的表尾[实际是先删除再插入]

TreeMap

TreeMap有如下特点

1:TreeMap实现了SortedMap接口

2:TreeMap和实现Map的其他对象不同的是,它不具有调优选项

3:TreeMap中所有元素必须是可比较的

4:TreeMap不是线程安全的

TreeMap的构造方法如下

1:TreeMap():构造一个空的TreeMap对象

2:TreeMap(Map m):构造一个具有和给定Map相同键-值对的TreeMap对象

3:TreeMap(Comparator c):构造一个空的TreeMap对象,并且规定特定的排列方式

4:TreeMap(SortedMap a):构造一个与给定的SortedMap具有相同键-值对,相同排列规则的TreeMap对象

TreeMap方法的简单使用

import java.util.*;
public class Warehouse1 {
    public static void main(String[] args) {
        TreeMap<String,String> treeMap=new TreeMap<>();
        treeMap.put("水果","苹果");
        treeMap.put("汽车","奔驰");
        treeMap.put("城市","西安");
        System.out.println("映射中的元素是:"+treeMap);
        Set<String> keySet=treeMap.keySet();
        Iterator<String> iterator=keySet.iterator();
        System.out.println("TreeMap类实现的Map映射,按键值升序排列元素如下:");
        while(iterator.hasNext()){
            String it=iterator.next();
            String name=treeMap.get(it);
            System.out.println(it+":"+name);
        }
    }
}

输出:

映射中的元素是:{城市=西安, 水果=苹果, 汽车=奔驰}
TreeMap类实现的Map映射,按键值升序排列元素如下:
城市:西安
水果:苹果
汽车:奔驰

注:TreeMap默认排序规则:按照key的字典顺序来排序[升序],当然也可以通过Comparator接口,去自定义排序规则。

HashMap和TreeMap的比较

HashMap和TreeMap在实际应用中要使用哪一个,还是需要根据实际情况进行分析,在Map中插入,删除和定位元素,HashMap是最好的选择,但是如果要按顺序遍历键,那么TreeMap会更好,根据集合大小,先把元素添加到HashMap,再把这种映射转换成一个用于有序键遍历的TreeMap会更快,使用HashMap要求添加的键类明确定义了hashcode()实现,有了TreeMap实现,添加到映射的元素一定是可排序的。

Hashtable

Hashtable有如下特点

1:Hashtable实现了map接口

2:Hashtable以Hash表基础构建

3:Hashtable和HashMap在执行过程中具有一定的相似性

4:Hashtable中不允许空元素,即键和值都不允许为空

5:Hashtable是线程安全的,可以在多线程环境中使用

Hashtable的四个构造方法如下

Hashtable();创建一个空Hashtable对象,并为其分配默认的初始容量和加载因子

Hashtable(int initialcapacity);创建一个空Hashtable对象,并为其分配指定的初始容量

Hashtable(int  initialcapacity,float loadFactor):创建一个空Hashtable对象,并指定其初始容量和加载因子

Hashtable(Map m);创建一个与给定Map对象具有相同键-值对的Hashtable对象

Hashtable新增的常用方法

举例:

import java.util.*;
public class Warehouse1 {
    public static void main(String[] args) {
    Hashtable<String,Double> hashtable=new Hashtable<>();
    hashtable.put("王俊凯",new Double(350.55));
        hashtable.put("王源",new Double(3250.55));
        hashtable.put("易烊千玺",new Double(1350.55));
        Enumeration<String> names=hashtable.keys();
        while(names.hasMoreElements()){
            String name=names.nextElement();
            System.out.println(name+":"+hashtable.get(name));
        }
        double new_double=hashtable.get("易烊千玺").doubleValue();
        hashtable.put("易烊千玺",new Double(new_double+2000));
        System.out.println("易烊千玺的新余额为:"+hashtable.get("易烊千玺"));
    }
}

输出:

易烊千玺:1350.55
王源:3250.55
王俊凯:350.55
易烊千玺的新余额为:3350.55

集合中元素的遍历

Collection接口的iterator()方法返回一个iterator对象,iterator接口能以迭代方式逐个访问集合中各个元素,并安全地从Collection中除去适当的元素,iterator接口中定义的方法如下所示:

iterator接口中的方法

  • boolean hasNext();判断游标右边是否还有元素
  • Object next();返回游标右边的元素并将游标移动到该元素后
  • void remove();删除游标左边的元素,通常在next()方法之后执行,只执行一次

Enumeration接口的功能与iterator接口类似,也能够对集合中的元素进行遍历,但是Enumeration接口只对Vector,Hashtable类提供遍历方法,并且不支持移除操作,实现Enumeration接口的对象,它生成一系列元素,一次生成一个,连续调用,nextElement方法将返回一系列的连续元素

Enumeration接口中的方法

boolean hasMoreELements();    如果存在访问的更多元素,则返回true

Object nextEment();        返回下一个元素的引用

到此这篇关于Java集合之Map接口与实现类详解的文章就介绍到这了,更多相关Java Map接口内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Java CompletableFuture实现原理分析详解

    Java CompletableFuture实现原理分析详解

    CompletableFuture是Java8并发新特性,本文我们主要来聊一聊CompletableFuture的回调功能以及异步工作原理是如何实现的,需要的可以了解一下
    2022-09-09
  • idea中方法、注释、导入类折叠或是展开的设置方法

    idea中方法、注释、导入类折叠或是展开的设置方法

    这篇文章主要介绍了idea中方法、注释、导入类折叠或是展开的设置,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2023-04-04
  • SpringMVC bean实现加载控制方法详解

    SpringMVC bean实现加载控制方法详解

    SpringMVC是一种基于Java,实现了Web MVC设计模式,请求驱动类型的轻量级Web框架,即使用了MVC架构模式的思想,将Web层进行职责解耦。基于请求驱动指的就是使用请求-响应模型,框架的目的就是帮助我们简化开发,SpringMVC也是要简化我们日常Web开发
    2022-08-08
  • Java实现在正则表达式中控制大小写的方法

    Java实现在正则表达式中控制大小写的方法

    这篇文章主要介绍了Java实现在正则表达式中控制大小写的方法,结合实例形式分析了java正则表达式中传递控制参数的功能与相关操作技巧,需要的朋友可以参考下
    2017-04-04
  • Spring Boot启动时调用自己的非web逻辑

    Spring Boot启动时调用自己的非web逻辑

    在spring Boot中,有些代码是WEB功能,例如API等,但是有些逻辑是非WEB,启动时就要调用并持续运行的,该如何加载自己的非WEB逻辑呢,下面通过实例代码给大家讲解,一起看看吧
    2017-07-07
  • mybatis中批量更新多个字段的2种实现方法

    mybatis中批量更新多个字段的2种实现方法

    当我们使用mybatis的时候,可能经常会碰到一批数据的批量更新问题,因为如果一条数据一更新,那每一条数据就需要涉及到一次数据库的操作,本文主要介绍了mybatis中批量更新多个字段的2种实现方法,感兴趣的可以了解一下
    2023-09-09
  • 以Spring Boot的方式显示图片或下载文件到浏览器的示例代码

    以Spring Boot的方式显示图片或下载文件到浏览器的示例代码

    这篇文章主要介绍了以Spring Boot的方式显示图片或下载文件到浏览器的示例代码,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-01-01
  • Spring Lifecycle的使用小结

    Spring Lifecycle的使用小结

    这篇文章主要介绍了Spring Lifecycle的使用,本文结合实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-05-05
  • SpringBoot结合FreeMarker视图渲染的实现

    SpringBoot结合FreeMarker视图渲染的实现

    FreeMarker它允许开发人员使用模板和数据来生成输出文本,如HTML网页、电子邮件、配置文件和源代码等,本文主要介绍了SpringBoot结合FreeMarker视图渲染的实现,感兴趣的可以了解一下
    2024-03-03
  • Java用单向环形链表来解决约瑟夫环Josepfu问题

    Java用单向环形链表来解决约瑟夫环Josepfu问题

    如果把单链表的最后一个节点的指针指向链表头部,而不是指向NULL,那么就构成了一个单向循环链表,通俗讲就是把尾节点的下一跳指向头结点
    2021-10-10

最新评论