资讯专栏INFORMATION COLUMN

Java8(4):当 forEach 需要索引

ckllj / 3266人阅读

摘要:比如运行结果那如果我们在遍历的时候需要使用到元素的索引呢类似方法那样很可惜,的并没有提供一个带索引的方法。的输入参数第一个即索引,第二个为元素。我们测试下这个方法运行结果结果和预期的一致。

在 上一篇文章 中,我们讨论了如何使用 Java8 中 Map 添加的新方法 computeIfAbsent 来统计集合中每个元素出现的所有位置,代码如下:

public static Map> getElementPositions(List list) {
    Map> positionsMap = new HashMap<>();

    for (int i = 0; i < list.size(); i++) {
        positionsMap.computeIfAbsent(list.get(i), k -> new ArrayList<>(1)).add(i);
    }

    return positionsMap;
}

至少有两点需要探讨:
1、如果 list 不是基于数组的(即不是 RandomAccess 的),而是基于链表的,那么 list.get(int index) 方法的效率就值得思考了;
2、既然都有了 Lambda(即当前平台为 Java8),我们为什么还要一次次去写传统的 for 循环呢?

在 Java8 中,为 Iterable 接口添加了默认的 forEach 方法:

很好理解,遍历当前 Iterable 中所有的元素,使用每个元素作为参数调用一次 action。而 Collection 接口继承了 Iterable 接口,所以所有的继承自 Collection 的集合类都可以直接调用 forEach 方法。比如:

public static void main(String[] args) throws Exception {
    List list = Arrays.asList("a", "b", "b", "c", "c", "c", "d", "d", "d", "f", "f", "g");
    
    list.forEach(str -> System.out.print(str + "  "));
    
    System.out.println();
}

运行结果:

那如果我们在遍历的时候需要使用到元素的索引呢(类似 getElementPositions 方法那样)?
很可惜,Java8 的 Iterable 并没有提供一个带索引的 forEach 方法。不过自己动手,丰衣足食 —— 让我们自己写一个带索引的 forEach 方法:

import java.util.Objects;
import java.util.function.BiConsumer;

/**
 * Iterable 的工具类
 */
public class Iterables {

    public static  void forEach(
            Iterable elements, BiConsumer action) {
        Objects.requireNonNull(elements);
        Objects.requireNonNull(action);

        int index = 0;
        for (E element : elements) {
            action.accept(index++, element);
        }
    }
}

forEach 方法第一个参数为要遍历的 Iterable,第二个参数为 BiConsumerBiConsumer 的输入参数第一个即索引,第二个为元素。

我们测试下这个 forEach 方法:

public static void main(String[] args) throws Exception {

    List list = Arrays.asList("a", "b", "b", "c", "c", "c", "d", "d", "d", "f", "f", "g");

    Iterables.forEach(list, (index, str) -> System.out.println(index + " -> " + str));
}

运行结果:

结果和预期的一致。

现在我们使用 Iterables.forEach 改写 getElementPositions 方法:

public static Map> getElementPositions(List list) {
    Map> positionsMap = new HashMap<>();

    Iterables.forEach(list, (index, str) -> {
        positionsMap.computeIfAbsent(str, k -> new ArrayList<>(1)).add(index);
    });

    return positionsMap;
}

public static void main(String[] args) throws Exception {
    List list = Arrays.asList("a", "b", "b", "c", "c", "c", "d", "d", "d", "f", "f", "g");

    System.out.println("使用 computeIfAbsent 和 Iterable.forEach:");
    Map> elementPositions = getElementPositions(list);
    System.out.println(elementPositions);
}

运行结果和原来一致:

真的不明白这么简单且实用的方法,Java8 为什么不在 Iterable 中提供一个默认实现(此处应有黑人问号)。

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/66372.html

相关文章

  • 乐字节-Java8核心特性实战之Stream(流)

    摘要:大家好,我是乐字节的小乐。需要注意的是很多流操作本身就会返回一个流,所以多个操作可以直接连接起来,如下图这样,操作可以进行链式调用,并且并行流还可以实现数据流并行处理操作。为集合创建并行流。 大家好,我是乐字节的小乐。说起流,我们会联想到手机、电脑组装流水线,物流仓库商品包装流水线等等,如果把手机 ,电脑,包裹看做最终结果的话,那么加工商品前的各种零部件就可以看做数据源,而中间一系列的...

    wenshi11019 评论0 收藏0
  • 乐字节-Java8新特性之Stream流(上)

    摘要:需要注意的是很多流操作本身就会返回一个流,所以多个操作可以直接连接起来,如下图这样,操作可以进行链式调用,并且并行流还可以实现数据流并行处理操作。为集合创建并行流。 上一篇文章,小乐给大家介绍了《Java8新特性之方法引用》,下面接下来小乐将会给大家介绍Java8新特性之Stream,称之为流,本篇文章为上半部分。 1、什么是流? Java Se中对于流的操作有输入输出IO流,而Jav...

    dingda 评论0 收藏0
  • Java 流程控制与数组

    摘要:静态初始化简化的语法格式动态初始化动态初始化只指定数组的长度,由系统为每个数组元素指定初始值。也就是说,数组引用变量是访问堆内存中数组元素的根本方式。 顺序结构 程序从上到下逐行地执行,中间没有任何判断和跳转。 分支结构 if条件语句 if语句使用布尔表达式或布尔值作为分支条件来进行分支控制。 第一种形式: if(logic expression) { ...

    DrizzleX 评论0 收藏0
  • 乐字节-Java8新特性-Lambda表达式

    摘要:很多语言等从设计之初就支持表达式。注意此时外部局部变量将自动变为作为方法返回值例子返回判断字符串是否为空判断字符串是否为空今天关于新特性表达式就讲到这里了,接下来我会继续讲述新特性之函数式接口。 上一篇文章我们了解了Java8新特性-接口默认方法,接下来我们聊一聊Java8新特性之Lambda表达式。 Lambda表达式(也称为闭包),它允许我们将函数当成参数传递给某个方法,或者把代码...

    gggggggbong 评论0 收藏0
  • Java 8 数据流教程

    摘要:数据流教程原文译者飞龙协议这个示例驱动的教程是数据流的深入总结。但是的数据流是完全不同的东西。数据流是单体,并且在函数式编程中起到重要作用。列表上的所有流式操作请见数据流的。基本的数据流使用特殊的表达式,例如,而不是,而不是。 Java 8 数据流教程 原文:Java 8 Stream Tutorial 译者:飞龙 协议:CC BY-NC-SA 4.0 这个示例驱动的教程是J...

    XUI 评论0 收藏0

发表评论

0条评论

最新活动
阅读需要支付1元查看
<