Java基础 集合(一)Collection与Iterator Iterable迭代器详解以及集合在for循环下如何删除

本文主要讨论了Java中的Iterator接口和Iterable接口的区别,以及在使用for循环遍历并删除集合元素时可能出现的问题,包括for-each循环下的并发修改异常和不同解决方案。
摘要由CSDN通过智能技术生成

目录

简介

Iterator和Iterable

for循环删除问题

for-each下删除数据

解决方法

for下删除数据

解决方法


前言-与正文无关

        生活远不止眼前的苦劳与奔波,它还充满了无数值得我们去体验和珍惜的美好事物。在这个快节奏的世界中,我们往往容易陷入工作的漩涡,忘记了停下脚步,感受周围的世界。让我们一起提醒自己,要适时放慢脚步,欣赏生活中的每一道风景,享受与家人朋友的温馨时光,发现那些平凡日子里隐藏的幸福时刻。因为,这些点点滴滴汇聚起来的,才是构成我们丰富多彩生活的本质。希望每个人都能在繁忙的生活中找到自己的快乐之源,不仅仅为了生存而工作,更为了更好的生活而生活。

        送你张美图!希望你开心!

简介

       集合可以看作是一种容器,用来存储对象信息。所有集合类都位于java.util包下,值得一提的是支持多线程的集合类位于java.util.concurrent包下。

        Java 中的 Collection 接口是 Java 集合框架的根接口之一。它提供了一组标准的方法,用于操作数据集合。Collection 接口本身不直接实现,而是通过各种具体的集合实现(如 ListSetQueue 等)。

集合总结:(以 JDK1.8 为例)

数据类型插入、删除时间复杂度查询时间复杂度底层数据结构是否线程安全
VectorO(N)O(1)数组是(已淘汰)
ArrayListO(N)O(1)数组
LinkedListO(1)O(N)双向链表
HashSetO(1)O(1)数组+链表+红黑树
TreeSetO(logN)O(logN)红黑树
LinkedHashSetO(1)O(1)~O(N)数组 + 链表 + 红黑树
ArrayDequeO(N)O(1)数组
PriorityQueueO(logN)O(logN)堆(数组实现)
HashMapO(1) ~ O(N)O(1) ~ O(N)数组+链表+红黑树
TreeMapO(logN)O(logN)数组+红黑树
HashTableO(1) / O(N)O(1) / O(N)数组+链表是(已淘汰)

Iterator和Iterable

在第一次看这两个接口,真以为是一模一样的,没发现里面有啥不同,存在即合理,它们两个还是有本质上的区别的。

首先来看Iterator接口:

public interface Iterator<E> {
    boolean hasNext();
    E next();
    void remove();
}

提供的API接口含义如下:

  • hasNext():判断集合中是否存在下一个对象,如果迭代器有更多的元素,则返回 true
  • next():返回集合中的下一个对象,并将访问指针移动一位
  • remove():删除集合中调用next()方法返回的对象

   Iterator 接口用于遍历集合(Collection)中的元素。它提供了一种标准的方法来逐个访问集合中的元素,而不需要了解集合的内部结构。

在早期,遍历集合的方式只有一种,通过Iterator迭代器操作

List<Integer> list = new ArrayList<>();
list.add(1);
list.add(2);
list.add(3);
//获取此集合的迭代器
Iterator iter = list.iterator();
// 遍历
while (iter.hasNext()) {
    // 获取遍历到当前下标的值
    Integer next = iter.next();
    System.out.println(next);
    if (next == 2) { 
        //删除数据
        iter.remove(); 
    }
}

再来看看Iterable 接口

Iterable是对Iterator的封装,在JDK 1.8时,实现了Iterable接口的集合可以使用增强 for 循环遍历集合对象,我们通过反编译后发现底层还是使用Iterator迭代器进行遍历,其实说白了他还是迭代器也是我们常用的for-each 循环。

请往下看。

public interface Iterable<T> {
    Iterator<T> iterator();
    // JDK 1.8
    default void forEach(Consumer<? super T> action) {
        Objects.requireNonNull(action);
        for (T t : this) {
            action.accept(t);
        }
    }
}

可以看到Iterable接口里面提供了Iterator接口,所以实现了Iterable接口的集合依旧可以使用迭代器遍历和操作集合中的对象;

而在 JDK 1.8中,Iterable提供了一个新的方法forEach(),它允许使用增强 for 循环遍历对象。

Iterable 接口的集合类可以通过 for-each 循环进行迭代。

List<Integer> list = new ArrayList<>();
for (Integer num : list) {
    System.out.println(num);
}

我们通过命令:javap -c反编译上面的这段代码后,发现它只是 Java 中的一个语法糖,本质上还是调用Iterator去遍历

翻译成代码,就和一开始的Iterator迭代器遍历方式基本相同了。

Iterator iter = list.iterator();
while (iter.hasNext()) {
    Integer num = iter.next();
    System.out.println(num);
}

for循环删除问题

使用场景: 当你需要遍历一个集合并可能在遍历过程中删除元素时,你需要使用 Iterator或者特殊的for循环,你要知道在ArrayList的for循环中是不被允许的。

for-each下删除数据

执行下述代码

public class Test01 {

    public static void main(String[] args) {
        List<String> list = new ArrayList<>();
        list.add("shuaige");
        list.add("laotou");
        list.add("meinv");
        list.add("shuaige");
        list.add("shuaige");

        for (String s : list) {
            if (s.equals("shuaige")) {
                list.remove(s);
            }

        }
        System.out.println(list);

    }
}

删除会报ConcurrentModificationException错,list发生数量变化,原因是:在单线程环境中,当迭代器创建后,除了迭代器自己的 remove 方法外,还有其他方式对集合进行结构上的修改。说白了for-each循环就是迭代器,迭代器不允许直接操作list删除数据!

解决方法

迭代器删除法

 public static void main(String[] args) {
        List<String> list = new ArrayList<>();
        list.add("shuaige");
        list.add("laotou");
        list.add("meinv");
        list.add("shuaige");
        list.add("shuaige");
        Iterator<String> iterator = list.iterator();
        while (iterator.hasNext()){
            if (iterator.next().equals("shuaige")){
                iterator.remove();
            }

        }

        System.out.println(list);

    }

for下删除数据

正常for循环不会报错但是值会删错,但是结果不是我们想要的结果,

 public static void main(String[] args) {
        List<String> list = new ArrayList<>();
        list.add("shuaige");
        list.add("laotou");
        list.add("meinv");
        list.add("shuaige");
        list.add("shuaige");
        for (int i = 0; i < list.size(); i++) {
            if (list.get(i).equals("shuaige")) {
                list.remove(i);
            }
        }

        System.out.println(list);

    }

结果:

原因:因为在list通过for循环遍历中删除元素之后,后面的元素会上前替补前面元素的位置,所以会漏掉一些值遍历进而有些值无法删除.

解决方法

倒序删除

    public static void main(String[] args) {
        List<String> list = new ArrayList<>();
        list.add("shuaige");
        list.add("laotou");
        list.add("meinv");
        list.add("shuaige");
        list.add("shuaige");
        for (int i = list.size()-1; i >= 0; i--) {
            if (list.get(i).equals("shuaige")) {
                list.remove(i);
            }
        }

        System.out.println(list);

    }

------------------------------------------与正文内容无关------------------------------------
 如果觉的文章写对各位读者老爷们有帮助的话,麻烦点赞加关注呗!作者在这拜谢了!

混口饭吃了!如果你需要Java 、Python毕设、商务合作、技术交流、就业指导、技术支持度过试用期。请在关注私信我,本人看到一定马上回复!

这是我全部文章所在目录,看看是否有你需要的,如果遇到觉得不对地方请留言,看到后我会查阅进行改正。

A乐神-CSDN博客

关注在文章左上角,作者信息处。

  • 21
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

A乐神

恭喜发财啊,老板,嘻嘻!

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值