Interview系列 - 05 Java|Iterator迭代器|集合继承体系|Set List Map接口特性|List实现类区别

news2024/12/23 9:25:59

文章目录

      • 01. 迭代器 Iterator 是什么?
      • 02. 迭代器 Iterator 有什么特点?
      • 03. 迭代器 Iterator 怎么使用?
      • 04. 如何边遍历边移除 Collection 中的元素?
      • 05. Iterator 和 ListIterator 有什么区别?
      • 06. 数组和集合的区别?
      • 07. 常见集合类的继承体系?
      • 08. 常见集合类的特点?
      • 09. 常见集合类的底层数据结构?
      • 10. Collection 和 Collections 有什么区别?
      • 11. List、Set、Map 之间的区别是什么?
      • 12. 怎么确保一个集合不能被修改?
      • 13. ArrayList 和 Vector 的区别?
      • 14. ArrayList 和 LinkedList 的区别?
      • 15. ArrayList,LinkedList,Vector的使用场景?
      • 16. Array 和 ArrayList 有何区别?
      • 17. 如何实现数组和 List 之间的转换?

01. 迭代器 Iterator 是什么?

Iterator则主要用于遍历Collection集合中的元素,Iterator对象也被称为迭代器。

02. 迭代器 Iterator 有什么特点?

Iterator 迭代器的特点是更加安全,它可以保证当使用Iterator迭代访问Collection集合元素时,Collection集合里的元素不能被改变,只有通过Iterator的remove方法删除上一次next方法返回的集合元素才可以;否则将会引发java.util.Concurrent ModificationException异常。

Iterator迭代器采用的是快速失败(fail-fast)机制,一旦在迭代过程中检测到该集合已经被修改(通常是程序中的其他线程修改),程序立即引发ConcurrentModificationException异常,这样可以避免共享资源而引发的潜在问题。

03. 迭代器 Iterator 怎么使用?

Iterator接口隐藏了各种Collection实现类的底层细节,向应用程序提供了遍历Collection集合元素的统一编程接口。
在这里插入图片描述

使用iterator()方法要求容器返回一个Iterator 对象:

(1) 当创建指向集合的 Iterator 对象后,指针指向第一个元素的上方,即指向一个空;

(2) 调用 hasNext() 方法:判断集合中是否还有元素;

(3) 调用 next() 方法:向下移动指针并返回指针指向的元素,如果指针指向的内存中没有元素,会报异常;

(4) 调用 remove() 方法:一般和next()方法一起用,用来删除next()方法返回的元素;

04. 如何边遍历边移除 Collection 中的元素?

① 《阿里巴巴编码规范》中的一条:不要在foreach循环里进行元素的remove/add操作,使用foreach循环进行删除则会抛出ConcurrentModificationException异常。

public class Main {
    public static void main(String[] args) {
        DepartmentMemberInput departmentMemberInput1 
            = new DepartmentMemberInput("张三","18767880909");
        DepartmentMemberInput departmentMemberInput2 
            = new DepartmentMemberInput("张四","18767880910");

        List<DepartmentMemberInput> list1 = new ArrayList<>();
        list1.add(departmentMemberInput1);
        list1.add(departmentMemberInput2);

        for (DepartmentMemberInput departmentMemberInput : list1) {
            if(departmentMemberInput.getName().equals("张三")){
                list1.remove(departmentMemberInput);
            }
        }
    }
}

执行结果:

在这里插入图片描述

② 不要在 iterator 遍历的过程中使用 list.remove() 方法:

public class Main {
    public static void main(String[] args) {
        DepartmentMemberInput departmentMemberInput1 
            = new DepartmentMemberInput("张三","18767880909");
        DepartmentMemberInput departmentMemberInput2 
            = new DepartmentMemberInput("张四","18767880910");

        List<DepartmentMemberInput> list1 = new ArrayList<>();
        list1.add(departmentMemberInput1);
        list1.add(departmentMemberInput2);

        for (Iterator iterator = list1.iterator(); iterator.hasNext();) {
            DepartmentMemberInput departmentMemberInput = (DepartmentMemberInput) iterator.next();
            if(departmentMemberInput.getName().equals("张三")){
                // 使用Iterator迭代过程中,不可修改集合元素,下面代码引发异常
                list1.remove(departmentMemberInput);
            }
        }
    }
}
public class IteratorErrorTest
{
    public static void main(String[] args)
    {
        // 创建一个集合
        Collection books=new HashSet();
        books.add("轻量级Java EE企业应用实战");
        books.add("Java讲义");
        // 获取books集合对应的迭代器
        Iterator it=books.iterator();
        while(it.hasNext())
        {
            String book=(String)it.next();
            System.out.println(book);
            if (book.equals("疯狂Android讲义"))
            {
                //使用Iterator迭代过程中,不可修改集合元素,下面代码引发异常
                books.remove(book);
            }
        }
    }
}

执行结果:

在这里插入图片描述

③ 推荐在 iterator 遍历的过程中使用iterator.remove() 方法:

public class Main {
    public static void main(String[] args) {
        DepartmentMemberInput departmentMemberInput1 
            = new DepartmentMemberInput("张三","18767880909");
        DepartmentMemberInput departmentMemberInput2 
            = new DepartmentMemberInput("张四","18767880910");

        List<DepartmentMemberInput> list1 = new ArrayList<>();
        list1.add(departmentMemberInput1);
        list1.add(departmentMemberInput2);

        for (Iterator iterator = list1.iterator(); iterator.hasNext();) {
            DepartmentMemberInput departmentMemberInput = (DepartmentMemberInput) iterator.next();
            if(departmentMemberInput.getName().equals("张三")){
                iterator.remove();
            }
        }
    }
}

05. Iterator 和 ListIterator 有什么区别?

(1) Iterator可用来遍历Set和List集合,但是ListIterator只能用来遍历List。

(2) Iterator对集合只能是前向遍历,ListIterator既可以前向也可以后向。

(3) ListIterator实现了Iterator接口,并包含其他的功能,比如:增加元素,替换元素,获取前一个和后一个元素的索引等等。

06. 数组和集合的区别?

数组元素既可以是基本类型的值,也可以是对象(实际上保存的是对象的引用变量);而集合里只能保存对象(实际上只是保存对象的引用变量,但通常习惯上认为集合里保存的是对象)。

07. 常见集合类的继承体系?

Java的集合类主要由两个接口派生而出:Collection和Map,Collection和Map是Java集合框架的根接口,这两个接口又包含了一些子接口或实现类。

在这里插入图片描述

Set和List接口是Collection接口派生的两个子接口,它们分别代表了无序集合和有序集合;Queue是Java提供的队列实现,有点类似于List。

在这里插入图片描述

08. 常见集合类的特点?

在这里插入图片描述

在这里插入图片描述

09. 常见集合类的底层数据结构?

集合实现类集合底层数据结构是否线程安全
ArrayList数组非线程安全
LinkedList双向链表非线程安全
Vector数组线程安全,效率较低,使用少
HashSetHashMap非线程安全
TreeSetTreeMap非线程安全
HashMap哈希表非线程安全
HashTable哈希表线程安全,所有的方法都有synchronized关键字,效率较低,使用少
PropertiesHashMap线程安全,key和value只能存储String字符串
TreeMap红黑树非线程安全

10. Collection 和 Collections 有什么区别?

Collection接口是List、Set和Queue接口的父接口,该接口里定义的方法既可用于操作Set集合,也可用于操作List和Queue集合。

Collections是一个操作Set、List和Map等集合的工具类,该工具类里提供了大量方法对集合元素进行排序、查询和修改等操作,还提供了将集合对象设置为不可变、对集合对象实现同步控制等方法。

常用集合框架中的实现类HashSet、TreeSet、ArrayList、ArrayDeque、LinkedList、HashMap和TreeMap都是线程不安全的。如果有多个线程访问它们,而且有超过一个的线程试图修改它们,则可能出现错误。Collections提供了多个synchronizedXxx()方法可以把它们包装成线程同步的集合。

public class SynchronizedTest{
    public static void main(String[] args){
        //下面程序创建了4个同步的集合对象
        Collection c=Collections.synchronizedCollection(new ArrayList());
        List list=Collections.synchronizedList(new ArrayList());
        Set s=Collections.synchronizedSet(new HashSet());
        Map m=Collections.synchronizedMap(new HashMap());
    }
}

11. List、Set、Map 之间的区别是什么?

List:一个有序(元素存入集合的顺序和取出的顺序一致)容器,元素可以重复,可以插入多个null元素,元素都有索引。常用的实现类有 ArrayList、LinkedList 和 Vector;

Set:一个无序(存入和取出顺序有可能不一致)容器,不可以存储重复元素,只允许存入一个null元素,必须保证元素唯一性。Set 接口常用实现类是 HashSet、LinkedHashSet 以及 TreeSet;

Map是一个键值对集合,存储键、值和之间的映射。 Key无序,唯一;value 不要求有序,允许重复。Map 的常用实现类:HashMap、TreeMap、HashTable、LinkedHashMap、ConcurrentHashMap;

img

12. 怎么确保一个集合不能被修改?

Collections提供了如下三类方法来返回一个不可变的集合:

(1) emptyXxx():返回一个空的、不可变的集合对象,此处的集合可以是List,Set,Map。

(2) singletonXxx():返回一个只包含指定对象(只有一个或一项元素)的、不可变的集合对象,此处的集合可以是List,Set,Map。

(3) unmodifiableXxx:返回指定集合对象的不可变视图,此处的集合可以是List,Set,Map。

上面三类方法的参数是原有的集合对象,返回值是该集合的“只读”版本。通过Collections提供的三类方法,可以生成“只读”的Collection或Map。

public class UnmodifiableTest{
    public static void main(String[] args){
        // 创建一个空的、不可改变的List对象
        List unmodifiableList=Collections.emptyList();
        // 创建一个只有一个元素,且不可改变的Set对象
        Set unmodifiableSet=Collections.singleton("Java讲义");
        // 创建一个普通的Map对象
        Map scores=new HashMap();
        scores.put("语文" , 80);
        scores.put("Java" , 82);
        //返回普通的Map对象对应的不可变版本
        Map unmodifiableMap=Collections.unmodifiableMap(scores);
        // 下面任意一行代码都将引发UnsupportedOperationException异常
        unmodifiableList.add("测试元素");    //①
        unmodifiableSet.add("测试元素");     //②
        unmodifiableMap.put("语文" , 90);   //③
    }
}

13. ArrayList 和 Vector 的区别?

ArrayList 和 Vector 均是 List 有序集合接口的实现类,都提供按照位置进行定位、添加或者删除的操作。

ArrayList是线程不安全的,当多个线程访问同一个ArrayList集合时,如果有超过一个线程修改了ArrayList集合,则程序必须手动保证该集合的同步性;但Vector集合则是线程安全的,无须程序保证该集合的同步性。因为Vector是线程安全的,所以Vector的性能比ArrayList的性能要低。实际上,即使需要保证List集合线程安全,也同样不推荐使用Vector实现类。可以使用Collections工具类提供的 List list = Collections.synchronizedList(new LinkedList());

14. ArrayList 和 LinkedList 的区别?

LinkedList与ArrayList、ArrayDeque的实现机制完全不同,ArrayList、ArrayDeque内部以动态数组的形式来保存集合中的元素,因此随机访问集合元素时有较好的性能;LinkedList内部以链表的形式来保存集合中的元素,因此随机访问集合元素时性能较差,但在插入、删除元素时性能非常出色(只需改变指针所指的地址即可)。需要指出的是,虽然Vector也是以数组的形式来存储集合元素的,但因为它实现了线程同步功能,所以各方面性能都有所下降。除此之外,LinkedList还实现了Deque接口,因此它可以被当成双端队列来使用,自然也可以被当成“栈”来使用了。

15. ArrayList,LinkedList,Vector的使用场景?

如果需要遍历List集合元素,对于ArrayList、Vector集合,应该使用随机访问方法(get)来遍历集合元素,这样性能更好;对于LinkedList集合,则应该采用迭代器(Iterator)来遍历集合元素。

如果需要经常执行插入、删除操作来改变List集合的大小,则应该使用LinkedList集合,而不是ArrayList。使用ArrayList、Vector集合需要经常重新分配内部数组的大小,其时间开销常常是使用LinkedList的时间开销的几十倍,效果很差。

如果有多个线程需要同时访问List集合中的元素,可考虑使用Collections将集合包装成线程安全的集合。

16. Array 和 ArrayList 有何区别?

(1) Array可以包含基本类型和对象类型,ArrayList只能包含对象类型。

(2) Array大小是固定的,定义后就不可被改变,ArrayList的大小是可变的。

(3) ArrayList提供了更多的方法和特性,比如:addAll(),removeAll(),iterator()等 。

17. 如何实现数组和 List 之间的转换?

(1) List转为数组:

直接调用ArrayList中的toArray方法就可以实现,List接口中,toArray有两个重载的方法:

Object[] toArray();
<T> T[] toArray(T[] a);

可见toArray可以用无入参的方式调用,返回一个Object数组;也可以用指定返回类型的方式调用,返回一个指定类型的数组。

@Test
public void test() {
    List<Object> rawList = new ArrayList();
    rawList.add("0");
    rawList.add("1");
    String[] arr = rawList.toArray(new String[rawList.size()]);
    for(String each:arr) {
        System.out.println(each);
    }
}

(2) 数组转换为 List:

如果需要将数组转为List,只需要使用Arrays.asList方法即可。

@Test
public void t5() {
    String[] arr = {"a","b","c"};
    List<String> list = Arrays.asList(arr);
    list.forEach(x -> System.out.println(x));
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/359087.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Alist ——本地网盘管理器

Alist ——本地网盘管理器 一、下载工具 Alist https://github.com/alist-org/alist二、启动登录 进入下载好的文件中&#xff0c;在地址栏输入cmd进入命令行启动 进入命令行输入 alist start启动 记住密码&#xff0c;和端口进入浏览器 输入 &#xff1a;127.0.0.1:5244用…

java final关键字 详解

概述&#xff1a;作用&#xff1a;细节&#xff1a;演示&#xff1a;总结&#xff1a;一、概述 : final [ˈ faɪnl]&#xff0c;最终的&#xff0c;最后的&#xff0c;决定性的&#xff0c;不可改变的。final作为Java中的一个关键字可以用来修饰类&#xff0c;方法&#xff0c…

【程序人生】从土木专员到网易测试工程师,薪资翻3倍,他经历了什么?

转行对于很多人来说&#xff0c;是一件艰难而又纠结的事情&#xff0c;或许缺乏勇气&#xff0c;或许缺乏魄力&#xff0c;或许内心深处不愿打破平衡。可对于我来说&#xff0c;转行是一件不可不为的事情&#xff0c;因为那意味着新的方向、新的希望。我是学工程管理的&#xf…

京东测试进阶之路:初入测试碎碎念篇

1、基本的测试用例设计方法 基本的测试用例设计方法&#xff08;边界值分析、等价类划分等&#xff09;。 业务和场景的积累&#xff0c;了解测试需求以及易出现的bug的地方。 多维角度设计测试用例&#xff08;用户、业务流程、异常场景、代码逻辑&#xff09;。 2、需求分析 …

idea自带maven位置、maven全局环境变量配置,安装jar到本地 mac

声明&#xff1a;本教程为mac版教程&#xff0c;Windows请路过 idea自带maven3配置全局环境变量 mac电脑maven3位置/Applications/IntelliJ\ IDEA.app/Contents/plugins/maven/lib/maven3配置全局变量,编~/.profile文件&#xff08;没有则新建&#xff09; export MAVEN/App…

JVM-JMM内存模型(happens-before、volatile)

前言 由于计算机的存储设备与处理器的运算速度有几个数量级的差距所以现代计算机系统都不得不加入一层读写速度尽可能接近处理器运算速度的高速缓存(Cache)来作为内存与处理器之间的缓冲。 将运算需要使用到的数据复制到缓存中&#xff0c;让运算能快速进行&#xff0c;当运算…

Day895.MySql误删数据还原方案 -MySQL实战

MySql误删数据还原方案 Hi&#xff0c;我是阿昌&#xff0c;今天学习记录的是关于MySql误删数据还原方案的内容。 传统的高可用架构是不能预防误删数据的&#xff0c;因为主库的一个 drop table 命令&#xff0c;会通过 binlog 传给所有从库和级联从库&#xff0c;进而导致整…

研报精选230220

目录 【行业230220国信证券】银行业行业专题&#xff1a;经济复苏中的优质中小银行【行业230220国信证券】汽车行业周报&#xff08;2023年第7周&#xff09;&#xff1a;吉利将发布新品牌“银河” &#xff0c;2022年宇通纯电动客车获欧洲销量冠军【行业230220开源证券】商贸零…

Java File相关操作

文章目录File文件操作IO流处理流缓冲流转换流对象流File文件操作 利用File类来操作。 文件操作中常用到相对目录和绝对路径 package org.File; import java.io.File; public class demo01 { public static void main(String[] args) { try{ File file new File("…

【Java期末复习】《面向对象程序设计》练习库

目录 一、单选题 二、填空题 三、程序填空题 1、 super使用--有如下父类和子类的定义&#xff0c;根据要求填写代码 2、简单加法计算器的实现 3、House类 4、矩形类 5、创建一个Box类&#xff0c;求其体积 四、函数题 6-1 求圆面积自定义异常类 6-2 判断一个数列是…

基于某业务单登陆场景并发测试实战

文章目录1 测试目的2 测试目标和测试对象3 名词解释4 测试说明5 测试环境和工具5.1 测试工具5.2 测试环境5.3 人力计划6 测试用例6.1 方案设计6.2 接口地址6.3 接口参数6.3.1 header参数6.3.2 请求参数7 脚本设计8 监控数据8.1 虚拟用户并发情况8.2 事务响应时间8.3 每秒点击次…

IP定位的具体网络应用

IP定位的原理是利用访问终端的注册信息和时延信息来估算现实中的地理位置&#xff0c;结合先进的定位算法服务&#xff0c;最高精确度能达到街道级&#xff0c;深受用户的喜爱。IP定位接口是一种通过IP地址来确定设备或用户位置的技术&#xff0c;主要应用于以下几个方面&#…

Windows平台上达梦数据库的ODBC安装与配置

文章目录概述安装包准备安装ODBC驱动配置ODBC数据源概述 最近很多公司都在响应信创&#xff0c;需要切换到国产数据库&#xff0c;然而很多数据库的一些基础组件的使用都没有一个很明确的官方文档。为了避免更多的人踩坑&#xff0c;本人将踩过的坑总结成博文&#xff0c;分享…

【正点原子FPGA连载】第十四章SD卡读写TXT文本实验 摘自【正点原子】DFZU2EG_4EV MPSoC之嵌入式Vitis开发指南

1&#xff09;实验平台&#xff1a;正点原子MPSoC开发板 2&#xff09;平台购买地址&#xff1a;https://detail.tmall.com/item.htm?id692450874670 3&#xff09;全套实验源码手册视频下载地址&#xff1a; http://www.openedv.com/thread-340252-1-1.html 第十四章SD卡读写…

网络小白入门之路--锐捷RGOS日常管理

锐捷作为网络界的一名重要成员&#xff0c;其设备的管理系统RGOS也是我们要学习的重要一环&#xff0c;那么今天我们就一起看下锐捷RGOS的日常管理。 锐捷设备日常登录方式&#xff1a; 本地登陆&#xff1a; Console登陆&#xff1a;全新或配置清空的设备&#xff0c;需要使…

TeX文件和md文件的转换

1 tex文件转换为md文件 1.1 安装 pandoc 首先安装 pandoc 到你的电脑 pandoc 1.2 命令行 可以在vscode终端输出入以下命令&#xff0c;命令运行路径为tex文件所在 cd tex文件所属的目录 pandoc -s tex文件名.tex -o md的文件名.md1.3预览 发现最后md文件只有一部分转化成…

171、【动态规划】leetcode ——309. 最佳买卖股票时机含冷冻期 (C++版本)

题目描述 原题链接&#xff1a;309. 最佳买卖股票时机含冷冻期 解题思路 本次难点在于多了一个冷冻期&#xff0c;大体的状态过程是三个&#xff1a;持有股票、不持有股票、冷冻期。在不持有股票时候&#xff0c;又可分为之前就不持有股票并继续保持、今天才卖出股票明天是冷…

视频直播美颜sdk的发展史

视频直播美颜sdk的出现是必然的&#xff0c;它的诞生就是为了让更多用户能够更好地体验到视频直播带给大家的快乐&#xff0c;可以说视频直播美颜sdk的发展和用户体验有着紧密联系。今天&#xff0c;小编不聊技术&#xff0c;咱们来聊一下视频直播美颜sdk的诞生以及发展历程。 …

抽奖动画大转盘抽奖思路与做法

抽奖是各类营销活动中最常见的一种形式&#xff0c;本产品需求大致如下&#xff1a;转盘周围跑马灯交替闪烁&#xff0c;点击抽奖&#xff0c;大转盘旋转&#xff0c;调用接口获取抽奖结果&#xff0c;大转盘指针指向对应的奖品。高保如下图12.整体思路本需求要求跑马灯交替闪烁…

PHP Composer 下载依赖项时删除其它不相关文件 导致项目报错 依赖包vendor手动导入

场景 最近项目上需要开发图片文字识别功能&#xff0c;接入了开源的tesseract_ocr&#xff0c;根据官网介绍&#xff0c;使用composer require thiagoalessio/tesseract_ocr命令安装依赖&#xff0c;但是在执行之后&#xff0c;删除了一些文件&#xff0c;导致项目报错。 解…