详解Java8的forEach(...)如何提供index值

网友投稿 434 2022-10-31


详解Java8的forEach(...)如何提供index值

java2遍历集合

遍历Collection的代码,可以是采用Iterator接口,通过next()遍历。如:

List list = Arrays.asList("Hi", "I", "am", "Henry.Yao");

// 此处已经用到了泛型,不能算是纯粹的Java2代码,仅作Iterator示范

for (Iterator it = list.iterator(); it.hasNext();) {

String item = it.next();

System.out.println("listItem = " + item);

}

输出:

listItem = Hi

listItem = I

listItem = am

listItem = Henry.Yao

Java5遍历集合

在Java5中,提供了增强的for循环,如:

List sFhuCjXaDvlist = Arrays.asList("Hi", "I", "am", "Henry.Yao");

for(String item : list) {

System.out.println("listItem = " + item);

}

Java8遍历集合

在Java8中,通过Lambda表达式提供了更简洁的编程方式,如:

list.forEach(item -> {

System.out.println("listItem = " + item);

});

需同时提供index,咋办?

操作集合元素item的同时,如果还需要同时提供index值,咋办?

思考后,我们可能大都写出了如下的代码,同时心有不甘:

List list = Arrays.asList("Hi", "I", "am", "Henry.Yao");

for(int index; index

String item = list.get(i);

System.out.println("list["+index+"] = "+item);

}

输出:

list[0] = Hi,

list[1] = I

list[2] = am

list[3] = Henry.Yao

期望的遍历模式

因为,如下的模式才是我们期望的模式

list.forEach((item, index) -> {

System.out.println("listItem = " + item);

}); // Compile ERROR

这只是期望。实际上,Jdk8并没有提供该函数,直至Jdk11也均没有提供该函数。

通过BiConsumer包装Consumer实现

“没有工具,我们制造工具” 定义如下的工具方法,基于这个工具方法,我们就能在遍历集合,同时提供item和index值:

// 工具方法

public static Consumer consumerWithIndex(BiConsumer consumer) {

class Obj {

int i;

}

Obj obj = new Obj();

return t -> {

int index = obj.i++;

consumer.accept(t, index);

};

}

这样的业务代码,是我期望的!

基于该工具方法,便可轻松编写如下业务代码,清晰、简洁:

list.forEach(LambdaUtils.consumerWithIndex((item, index) -> {

System.out.println("list[" + index + "]=" + item);

}));

思考过程

这个工具方法的设计过程,也是参考借鉴了distinctByKey,如图:

// 工具方法

public static Predicate distinctByKey(Function super T, ?> keyExtractor) {

Map seen = new ConcurrentHashMap<>();

return t -> Objects.isNull(seen.putIfAbsent(keyExtractor.apply(t), Boolean.TRUE));

}

// 业务代码

// 从人员列表中过滤出一个子集(每个部门选一个人)

employees.stream().filter(distinctByKey(Employee::getDeptCode)).collect(toList());

我们不仅要会使用工具,更要会制造工具…

我们的程序,不仅仅只是大片的业务代码http://,更是需要抽象和提取出的诸多工具方法。

使用工具(使用Java和第三方提供的方法)到极致,那是高级程序员,

制造工具(分析和设计出项目的工具方法)到极致,那是高级设计师。

String item = list.get(i);

System.out.println("list["+index+"] = "+item);

}

输出:

list[0] = Hi,

list[1] = I

list[2] = am

list[3] = Henry.Yao

期望的遍历模式

因为,如下的模式才是我们期望的模式

list.forEach((item, index) -> {

System.out.println("listItem = " + item);

}); // Compile ERROR

这只是期望。实际上,Jdk8并没有提供该函数,直至Jdk11也均没有提供该函数。

通过BiConsumer包装Consumer实现

“没有工具,我们制造工具” 定义如下的工具方法,基于这个工具方法,我们就能在遍历集合,同时提供item和index值:

// 工具方法

public static Consumer consumerWithIndex(BiConsumer consumer) {

class Obj {

int i;

}

Obj obj = new Obj();

return t -> {

int index = obj.i++;

consumer.accept(t, index);

};

}

这样的业务代码,是我期望的!

基于该工具方法,便可轻松编写如下业务代码,清晰、简洁:

list.forEach(LambdaUtils.consumerWithIndex((item, index) -> {

System.out.println("list[" + index + "]=" + item);

}));

思考过程

这个工具方法的设计过程,也是参考借鉴了distinctByKey,如图:

// 工具方法

public static Predicate distinctByKey(Function super T, ?> keyExtractor) {

Map seen = new ConcurrentHashMap<>();

return t -> Objects.isNull(seen.putIfAbsent(keyExtractor.apply(t), Boolean.TRUE));

}

// 业务代码

// 从人员列表中过滤出一个子集(每个部门选一个人)

employees.stream().filter(distinctByKey(Employee::getDeptCode)).collect(toList());

我们不仅要会使用工具,更要会制造工具…

我们的程序,不仅仅只是大片的业务代码http://,更是需要抽象和提取出的诸多工具方法。

使用工具(使用Java和第三方提供的方法)到极致,那是高级程序员,

制造工具(分析和设计出项目的工具方法)到极致,那是高级设计师。


版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:索尼全画幅微单A7/A7R上市
下一篇:yii2 redis 常用命令
相关文章

 发表评论

评论列表