删除Map的中某一项报错

package part;
import java.util.HashMap;
import java.util.Set;
public class Java01 {
public static void main(String[] args) {
// 为啥这里要使用包装类型 Integer, 而不是int
HashMap<String, Integer> map = new HashMap();
map.put("a", 1);
map.put("b", 2);
map.put("c", 3);
// 获取所有键名的集合,返回类型是 Set 类型
Set<String> Keys = map.keySet();
// 增强型for循环
for (String keyName : Keys) {
// 键名相等的话,删除掉, 导致程序报错了呀。 会报错
if(keyName == "b"){
// map.remove(keyName); 删除会报错的
map.put("n",11); // 新增也是会报错的
map.put(keyName, 200); // ok的,但是我们最好还是使用迭代器来操作
}
}
}
}

为啥这里要使用包装类型 Integer, 而不是int

因为: HashMap的键和值必须是对象类型,不能是基本数据类型。

Java 提供了自动装箱(int 转 Integer)和拆箱(Integer 转 int)的功能

int 类型自动装箱后就是 Integer

为哈会报错呢?

我们在循环中途的某一项的时候,不光是删除,新增也会报错的

因为:当你使用 for-each 循环遍历 HashMap 的键集合时

for-each 底层是通过 Iterator 实现的

Iterator 会检查集合是否被修改(通过一个 modCount 变量)来判断

如果发现集合被修改(例添加、删除元素),就会抛出 ConcurrentModificationException

为啥在遍历最后一项的时候删除就不会报错呢


package part;
import java.util.HashMap;
import java.util.Set; public class Java01 {
public static void main(String[] args) {
// HashMap的键和值必须是对象类型,不能是基本数据类型。 int 类型自动装箱后就是 Integer
HashMap<String, Integer> map = new HashMap();
map.put("a", 1);
map.put("b", 2);
map.put("c", 3);
// 获取所有键名的集合,返回类型是 Set 类型
Set<String> Keys = map.keySet();
// 增强型for循环
for (String keyName : Keys) {
// 当遍历最后一项的时候,删除就不会报错
if(keyName == "c"){
map.remove(keyName);
}
}
}
}

解释:为什么删除最后一项不会报错?

这个跟(ConcurrentModificationException)触发机制

(ConcurrentModificationException) 是通过 modCount 变量来检测集合是否被修改的。

在遍历集合时,Iterator 会检查 modCount 是否与预期值一致。如果不一致(即集合被修改),就会抛出异常。

当你删除最后一项时,Iterator 可能已经完成了遍历,因此不会触发 modCount 的检查。

因此也就不会报错哈

修改数据最好使用迭代器来处理

package part;

import java.util.HashMap;
import java.util.Iterator;
import java.util.Set; public class Java01 {
public static void main(String[] args) {
// 为啥这里要使用包装类型 Integer, 而不是int
HashMap<String, Integer> map = new HashMap();
map.put("a", 1);
map.put("b", 2);
map.put("c", 3);
Set<String> keys = map.keySet();
// 迭代器
Iterator<String> it =keys.iterator();
// hasNext方法用于办法是否存在下一条数据
while (it.hasNext()) {
// 获取下一条数据
String key = it.next();
// 删除键名是b这一项
if("b".equals(key)){
it.remove();
}
// 1 null 3
System.out.println(map.get(key));
}
// {a=1, c=3}
System.out.println(map);
}
}

使用 removeIf() 方法(Java 8+)迭代

HashMap<String, Integer> map = new HashMap<>();
map.put("a", 1);
map.put("b", 2);
map.put("c", 3); Set<String> keys = map.keySet();
keys.removeIf(keyName -> keyName.equals("c")); // 使用 removeIf
System.out.println(map); // 输出: {a=1, b=2}

使用迭代器能够删除其他项吗?

就是说:我们在循环a这一项的时候,可以删除b这一项的数据吗?

不可以的。

因为:只能够删除当前循环的这一项的。再说一次:只能够删除当前循环的这一项的

将数组转化为字符串 Arrays.toString()


package part; import java.util.Arrays;
public class Java01 {
public static void main(String[] args) {
// 声明并初始化一个 int 数组
int[] is= {1, 2, 3, 4, 5};
// 转化为字符串 [1, 2, 3, 4, 5]
String str = Arrays.toString(is);
System.out.println(str);
// [I@28d93b30 是hashCode的内存地址
System.out.println(is);
}
}

将数组转化为集合以及升序

package part;

import java.util.Arrays;
import java.util.List; public class Java01 {
public static void main(String[] args) {
// 将数组转化为集合
List<Integer> list = Arrays.asList(1, 2, 3, 4, 5);
int[] arr = {4,1,-3,10 };
// Arrays.sort 默认是升序的哈,会影响原始数组的哈,与js不同的
Arrays.sort(arr);
// [-3, 1, 4, 10]
System.out.println(Arrays.toString(arr));
}
}

二分查找法,查找的是排序后的位置

package part;

import java.util.Arrays;
import java.util.List; public class Java01 {
public static void main(String[] args) {
int[] arr = {4,1,-3,10 };
Arrays.sort(arr);
// [-3, 1, 4, 10]
System.out.println(Arrays.toString(arr));
// 查询4的位置
int index = Arrays.binarySearch(arr, 4);
// 2
System.out.println(index);
}
}

2个数组项比较

package part;
import java.util.Arrays;
import java.util.List; public class Java01 {
public static void main(String[] args) {
int[] arr1 = {4,1,-3,10 };
int[] arr2 = {4,1,10, -3};
// 会比较2个数组是否相等,会一对一进行比较。 第2项的-3和第2项的10不相等,返回flase
System.out.println(Arrays.equals(arr1,arr2)); // false
}
}
package part;

import java.util.Arrays;
import java.util.List; public class Java01 {
public static void main(String[] args) {
int[] arr1 = {4, 1, -3, 10};
int[] arr2 = {4, 1, 10, -3};
//arr1, 0, 2表示从arr1数组中,从0开始取,取前2个。 arr2, 0, 2表示从arr2数组中,从0开始取,取前2个
// 特别需要注意一点的是: 是在 JDK 9 中引入的。如果你使用的是 JDK 8 或更早版本,这个方法会报错。
System.out.println(Arrays.equals(arr1, 0, 2, arr2, 0, 2)); // 输出: true
}
}

ArrayList 集合默认时长度10,你设置容器大小必须大于等于0。如果是负数会报错


package part; import java.util.ArrayList;
public class Java01 {
public static void main(String[] args) {
ArrayList<String > list2 = new ArrayList<String>(0);
System.out.println(list2); // [] ArrayList<String> list1 = new ArrayList<String>(-1);
System.out.println(list1); // ava.lang.IllegalArgumentException 非法参数异常
}
}

ArrayList的访问范围是[0,长度-1]

package part;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List; public class Java01 {
public static void main(String[] args) {
ArrayList list = new ArrayList();
list.add("A");
list.add("B");
list.add("C");
// 访问的范围是[0,长度-1]
System.out.println(list.get(2));
// IndexOutOfBoundsException 索引超出异常
System.out.println(list.get(3));
}
}

LinkedList的长度是0,使用get(0)和getFirst访问报错

package part;

import java.util.LinkedList;

public class Java01 {
public static void main(String[] args) {
LinkedList<String> list = new LinkedList<String>();
//报错 IndexOutOfBoundsException
System.out.println(list.get(0));
// list的长度是0, 现在你获取第1项,这样的操作跟索引越界相似。同样也是会报错的
// java.util.NoSuchElementException
System.out.println(list.getFirst());
}
}

HashMap在循环的时候删除,新增数据会报错


package part;
import java.util.HashMap;
import java.util.LinkedList; public class Java01 {
public static void main(String[] args) {
HashMap map = new HashMap();
map.put("a",1);
map.put("b",2);
map.put("c",3);
// HashMap一旦开始循环,那么如果删除,新增数据,就会发生错误
for (Object key : map.keySet()) {
if(key.equals("b")){
// map.remove(key); 删除数据报错
// map.put("d",4); 新增数据报错
map.put(key,4); // 修改数据不会报错的哈
}
}
// 我们可以使用迭代器还解决这样的问题哈。
System.out.println(map);
}
}

尾声

准备开始学习java了。

今天学习的第五天,每天都会发文章,我要卷起来。

请小伙伴们监督我,奥利给

java集合中的迭代器Iterator和数组内置方法以及常见的报错的更多相关文章

  1. 0513JS数组内置方法、数学函数、时间函数

    |数组中常用的内置方法|-push()与pop()|--push()是往数组的尾部添加,同时返回新数组的长度 var attr = [1,2,3,4,5];var attr2 = [6,7,8,9,0 ...

  2. JAVA集合中的迭代器的遍历

    JAVA中的迭代器,迭代实质上就是遍历,在JAVA中使用iterator()方法进行迭代.需要注意的是,iterator()方法的返回值是Iterator对象.Iterator对象有三个方法,hasN ...

  3. 【Python】Java程序员学习Python(四)— 内置方法和内置变量

    <假如爱有天意> 当天边那颗星出现,你可知我又开始想念,有多少爱恋只能遥遥相望,就像月光洒向海面,年少的我们曾以为,相爱的人就能到永远,当我们相信情到深处在一起,听不见风中的叹息,谁知道爱 ...

  4. js中数组内置方法

    var arr = ['A','B','C','D']; length 计算数组的长度 arr.length//4 indexOf() 搜索一个指定的元素的位置 arr.indexOf('C');// ...

  5. js数组内置方法

    var arr = ['A','B','C','D']; length 计算数组的长度 arr.length//4   indexOf() 搜索一个指定的元素的位置 arr.indexOf('C'); ...

  6. 自己封装函数,实现数组的内置方法indexOf的功能

    在学习或开发过程中,经常会有朋友需要使用到一个数组方法-indexOf,这里我们先来谈谈它的功能:返回指定数据所在的索引,如果没有则返回-1. 那么我们在使用时通常是直接使用它这个数组内置方法 今天这 ...

  7. 用JAVA编写浏览器内核之实现javascript的document对象与内置方法

    原创文章.转载请注明. 阅读本文之前,您须要对浏览器怎样载入javascript有一定了解. 当然,对java与javascript本身也须要了解. 本文首先介绍浏览器载入并执行javascript的 ...

  8. 牛客网Java刷题知识点之Java 集合框架的构成、集合框架中的迭代器Iterator、集合框架中的集合接口Collection(List和Set)、集合框架中的Map集合

    不多说,直接上干货! 集合框架中包含了大量集合接口.这些接口的实现类和操作它们的算法. 集合容器因为内部的数据结构不同,有多种具体容器. 不断的向上抽取,就形成了集合框架. Map是一次添加一对元素. ...

  9. Java集合中迭代器的常用用法

    该例子展示了一个Java集合中迭代器的常用用法public class LinkedListTest { public static void main(String[] args) { List&l ...

  10. Java集合中Set的常见问题及用法

    在这里演示的案例是衔接Java集合中的List(点击查看)那篇博文的,本节我们学习的Set的用法. Set是Collection的一个重要的子接口,Set中的元素是无序排列的,并且元素不可以重复,被称 ...

随机推荐

  1. Open-RAG:将开源LLM模型集成为高效RAG模型 | ENMLP'24

    本文是对公开论文的核心提炼,旨在进行学术交流.如有任何侵权问题,请及时联系号主以便删除. 来源:晓飞的算法工程笔记 公众号,转载请注明出处 论文: Open-RAG: Enhanced Retriev ...

  2. 使用Acme.sh免费签发SSL证书

    github:https://github.com/acmesh-official/acme.sh 概述一个纯粹用Shell(Unix shell)语言编写的ACME协议客户端.完整的ACME协议实施 ...

  3. 【3分钟学会】一招禁用表单中input输入框回车键自动触发提交事件!

    知其然知其所以然 在前端项目开发中,偶尔会有表单提交的问题: 用户输入表单后,不小心按了回车键,导致提前触发了提交事件? 问题概述 当表单中仅有一个input输入框时,按下回车键就会自动触发提交事件, ...

  4. 终于解决了.net在线客服系统总是被360误报的问题(对软件进行数字签名)

    升讯威在线客服与营销系统是基于 .net core / WPF 开发的一款在线客服软件,宗旨是: 开放.开源.共享.努力打造 .net 社区的一款优秀开源产品. 背景 我在业余时间开发的这个客服系统, ...

  5. Winform在主窗体里切换多个窗体

    1.点击解决方案资源管理器的项目名称,右键添加用户控件(Windows窗体). 2.在主窗体代码中实例化添加的用户控件(Windows窗体). 点击查看代码 UserControl1 userCont ...

  6. JavaScript是按顺序执行的吗?聊聊JavaScript中的变量提升

    作为一位前端开发者,我们经常会听到这么一句话:"JavaScript的执行是按照顺序自上而下依次执行的."这句话说的并没有错.但是它似乎又好像不完全对.我们先来看以下这段代码.你觉 ...

  7. 零基础学习人工智能—Python—Pytorch学习(十二)

    前言 本文介绍使用神经网络进行实战. 使用的代码是<零基础学习人工智能-Python-Pytorch学习(九)>里的代码. 代码实现 mudule定义 首先我们自定义一个module,创建 ...

  8. openEuler欧拉部署gitbook

    安装nodejs10 参见<openEuler欧拉安装指定版本的nodejs> 安装Gitbook n 16 # 选择高版本的node npm config set registry ht ...

  9. Qt通用方法及类库8

    函数名 //异或加密算法 static QString getXorEncryptDecrypt(const QString &str, char key); //异或校验 static uc ...

  10. 在C++中实现委托事件的方法

    参考链接: 1.在C++中模拟委托事件的方法(一) 2.利用C++的模板模拟.net的代理语法 源码学习: 1.https://pan.baidu.com/s/15vbryvzDnvmJ6FMku6_ ...