您当前的位置:首页 > 电脑百科 > 程序开发 > 语言 > JAVA

Java小技巧:巧用函数方法实现二维数组遍历

时间:2022-08-07 16:05:59  来源:掘金  作者:一灰灰

对于数组遍历,基本上每个开发者都写过,遍历本身没什么好说的,但是当我们在遍历的过程中,有一些复杂的业务逻辑时,将会发现代码的层级会逐渐加深

如一个简单的case,将一个二维数组中的偶数找出来,保存到一个列表中

二维数组遍历,每个元素判断下是否为偶数,很容易就可以写出来,如

public void getEven() {    int[][] cells = new int[][]{{1, 2, 3, 4}, {11, 12, 13, 14}, {21, 22, 23, 24}};    List<Integer> ans = new ArrayList<>();    for (int i = 0; i < cells.length; i ++) {        for (int j = 0; j < cells[0].length; j++) {            if ((cells[i][j] & 1) == 0) {                ans.add(cells[i][j]);            }        }    }    System.out.println(ans);}复制代码

上面这个实现没啥问题,但是这个代码的深度很容易就有三层了;当上面这个if中如果再有其他的判定条件,那么这个代码层级很容易增加了;二维数组还好,如果是三维数组,一个遍历就是三层;再加点逻辑,四层、五层不也是分分钟的事情么

那么问题来了,代码层级变多之后会有什么问题呢?

只要代码能跑,又能有什么问题呢?!

1. 函数方法消减代码层级

由于多维数组的遍历层级天然就很深,那么有办法进行消减么?

要解决这个问题,关键是要抓住重点,遍历的重点是什么?获取每个元素的坐标!那么我们可以怎么办?

定义一个函数方法,输入的就是函数坐标,在这个函数体中执行我们的遍历逻辑即可

基于上面这个思路,相信我们可以很容易写一个二维的数组遍历通用方法

public static void scan(int maxX, int maxY, BiConsumer<Integer, Integer> consumer) {    for (int i = 0; i < maxX; i++) {        for (int j = 0; j < maxY; j++) {            consumer.accept(i, j);        }    }}复制代码

主要上面的实现,函数方法直接使用了JDK默认提供的BiConsumer,两个传参,都是int 数组下表;无返回值

那么上面这个怎么用呢?

同样是上面的例子,改一下之后,如

public void getEven() {    int[][] cells = new int[][]{{1, 2, 3, 4}, {11, 12, 13, 14}, {21, 22, 23, 24}};    List<Integer> ans = new ArrayList<>();    scan(cells.length, cells[0].length, (i, j) -> {        if ((cells[i][j] & 1) == 0) {            ans.add(cells[i][j]);        }    });    System.out.println(ans);}复制代码

相比于前面的,貌似也就少了一层而已,好像也没什么了不起的

但是,当数组变为三维、四维、无维时,这个改动的写法层级都不会变哦

2. 遍历中return支持

前面的实现对于正常的遍历没啥问题;但是当我们在遍历过程中,遇到某个条件直接返回,能支持么?

如一个遍历二维数组,我们希望判断其中是否有偶数,那么可以怎么整?

仔细琢磨一下我们的scan方法,希望可以支持return,主要的问题点就是这个函数方法执行之后,我该怎么知道是继续循环还是直接return呢?

很容易想到的就是执行逻辑中,添加一个额外的返回值,用于标记是否中断循环直接返回

基于此思路,我们可以实现一个简单的demo版本

定义一个函数方法,接受循环的下标 + 返回值

@FunctionalInterfacepublic interface ScanProcess<T> {    ImmutablePair<Boolean, T> accept(int i, int j);}

循环通用方法就可以相应地改成

public static <T> T scanReturn(int x, int y, ScanProcess<T> func) {    for (int i = 0; i < x; i++) {        for (int j = 0; j < y; j++) {            ImmutablePair<Boolean, T> ans = func.accept(i, j);            if (ans != null && ans.left) {                return ans.right;            }        }    }    return null;}复制代码

基于上面这种思路,我们的实际使用姿势如下

@Testpublic void getEven() {    int[][] cells = new int[][]{{1, 2, 3, 4}, {11, 12, 13, 14}, {21, 22, 23, 24}};    List<Integer> ans = new ArrayList<>();    scanReturn(cells.length, cells[0].length, (i, j) -> {        if ((cells[i][j] & 1) == 0) {            return ImmutablePair.of(true, i + "_" + j);        }        return ImmutablePair.of(false, null);    });    System.out.println(ans);}复制代码

上面这个实现可满足我们的需求,唯一有个别扭的地方就是返回,总有点不太优雅;那么除了这种方式之外,还有其他的方式么?

既然考虑了返回值,那么再考虑一下传参呢?通过一个定义的参数来装在是否中断以及返回结果,是否可行呢?

基于这个思路,我们可以先定义一个参数包装类

public static class Ans<T> {    private T ans;    private boolean tag = false;    public Ans<T> setAns(T ans) {        tag = true;        this.ans = ans;        return this;    }    public T getAns() {        return ans;    }}public interface ScanFunc<T> {    void accept(int i, int j, Ans<T> ans)}复制代码

我们希望通过Ans这个类来记录循环结果,其中tag=true,则表示不用继续循环了,直接返回ans结果吧

与之对应的方法改造及实例如下

public static <T> T scanReturn(int x, int y, ScanFunc<T> func) {    Ans<T> ans = new Ans<>();    for (int i = 0; i < x; i++) {        for (int j = 0; j < y; j++) {            func.accept(i, j, ans);            if (ans.tag) {                return ans.ans;            }        }    }    return null;}    public void getEven() {    int[][] cells = new int[][]{{1, 2, 3, 4}, {11, 12, 13, 14}, {21, 22, 23, 24}};    String ans = scanReturn(cells.length, cells[0].length, (i, j, a) -> {        if ((cells[i][j] & 1) == 0) {            a.setAns(i + "_" + j);        }    });    System.out.println(ans);}复制代码

这样看起来就比前面的要好一点了

实际跑一下,看下输出是否和我们预期的一致;

 


作者:一灰灰
链接:
https://juejin.cn/post/7128771846131941406



Tags:Java   点击:()  评论:()
声明:本站部分内容及图片来自互联网,转载是出于传递更多信息之目的,内容观点仅代表作者本人,如有任何标注错误或版权侵犯请与我们联系(Email:2595517585@qq.com),我们将及时更正、删除,谢谢。
▌相关推荐
1、原理:基于javaAgent和Java字节码注入技术的java探针工具技术原理 2、原理分析动态代理功能实现说明,我们利用javaAgent和ASM字节码技术开发java探针工具,实现原理如下:jdk1.5...【详细内容】
2022-10-24  Tags: Java  点击:(4)  评论:(0)  加入收藏
简单描述java虚拟机内存分配与GC触发场景堆内存中,新生代和老年代分区图解 堆空间的参数设置-XX: +PrintFlagsInitial :查看所有的参数的默认初始值-XX: +PrintFlagsFinal :...【详细内容】
2022-10-21  Tags: Java  点击:(5)  评论:(0)  加入收藏
Java强大的功能需要依托在软件工具上实现,所以Java编程开发需要用到编程工具,市面上有哪些Java编程工具呢?那种编程工具好呢?该怎么选择呢?接下来罗列了一系列的Java编程开发工具...【详细内容】
2022-10-17  Tags: Java  点击:(11)  评论:(0)  加入收藏
Java是一门用来开发软件的计算机语言,如同汉语是在中国进行交流你的语言,西班牙语是在西班牙交流的语言,编程自然适合计算机沟通,就需要使用计算机认识的语言,那么Java就是其中一...【详细内容】
2022-10-15  Tags: Java  点击:(6)  评论:(0)  加入收藏
相信很多学生 在 学习英语的时候都有这样那样的问题 ,因此 也 困扰着太多 想要学习编程的小伙伴。其实学习编程,常用的单词就那么多,只要把必备的单词学会,你的代码也能写的很6,...【详细内容】
2022-10-14  Tags: Java  点击:(12)  评论:(0)  加入收藏
前段时间有一个叫做“人类高质量男性”的视频火了,相信很多同学都刷到过。所以今天给大家分享下,什么叫做“人类高质量代码”,哈哈,开个玩笑。其实分享的都是一些自己平时总结的...【详细内容】
2022-10-11  Tags: Java  点击:(14)  评论:(0)  加入收藏
JVM 内存有以下区域: Young Generation Old Generation Metaspace Others region 图:JVM 内存区域 要查看哪些对象存储在哪个区域,您可以参考此视频剪辑。有时您的应用程序...【详细内容】
2022-10-04  Tags: Java  点击:(19)  评论:(0)  加入收藏
属性文件乱码通常是中文乱码,在英文下通常没有这个问题。 如上图显示的,中文字符在属性文件中读取后会显示为乱码.问题和解决导致这个问题的主要原因是属性文件如果你不进行设...【详细内容】
2022-10-04  Tags: Java  点击:(14)  评论:(0)  加入收藏
JoinerGuava Joiner顾名思义就是将字符串连接起来 Joiner joiner = Joiner.on("; ").skipNulls(); System.out.println(joiner.join("Harry", null, "Ron", "Hermione"));...【详细内容】
2022-10-03  Tags: Java  点击:(26)  评论:(0)  加入收藏
在创建 JavaScript 模块时,export 语句用于从模块中导出实时绑定的函数、对象或原始值,以便其他程序可以通过 import 语句使用它们。被导出的绑定值依然可以在本地进行修改。...【详细内容】
2022-09-23  Tags: Java  点击:(34)  评论:(0)  加入收藏
▌简易百科推荐
1、原理:基于javaAgent和Java字节码注入技术的java探针工具技术原理 2、原理分析动态代理功能实现说明,我们利用javaAgent和ASM字节码技术开发java探针工具,实现原理如下:jdk1.5...【详细内容】
2022-10-24  马士兵Java架构  今日头条  Tags:Java探针   点击:(4)  评论:(0)  加入收藏
简单描述java虚拟机内存分配与GC触发场景堆内存中,新生代和老年代分区图解 堆空间的参数设置-XX: +PrintFlagsInitial :查看所有的参数的默认初始值-XX: +PrintFlagsFinal :...【详细内容】
2022-10-21  chost-jie    Tags:java虚拟机   点击:(5)  评论:(0)  加入收藏
概述最近项目上反馈某个重要的定时任务突然不执行了,很头疼,开发环境和测试环境都没有出现过这个问题。定时任务采用的是 ScheduledThreadPoolExecutor,后来一看代码发现踩了一...【详细内容】
2022-10-21  Java架构学习指南  今日头条  Tags:   点击:(7)  评论:(0)  加入收藏
Java 流不能很好地处理已检查的异常。在本文中,深入探讨如何管理此类问题。 Java 引入了检查异常的概念。与早期的方法相比,强制开发人员管理异常的想法是革命性的。如今,Java...【详细内容】
2022-10-21  qaseven     Tags:Lambda   点击:(1)  评论:(0)  加入收藏
Java强大的功能需要依托在软件工具上实现,所以Java编程开发需要用到编程工具,市面上有哪些Java编程工具呢?那种编程工具好呢?该怎么选择呢?接下来罗列了一系列的Java编程开发工具...【详细内容】
2022-10-17  farsightCD  搜狐号  Tags:Java编程   点击:(11)  评论:(0)  加入收藏
Java是一门用来开发软件的计算机语言,如同汉语是在中国进行交流你的语言,西班牙语是在西班牙交流的语言,编程自然适合计算机沟通,就需要使用计算机认识的语言,那么Java就是其中一...【详细内容】
2022-10-15  添甄  今日头条  Tags:Java   点击:(6)  评论:(0)  加入收藏
相信很多学生 在 学习英语的时候都有这样那样的问题 ,因此 也 困扰着太多 想要学习编程的小伙伴。其实学习编程,常用的单词就那么多,只要把必备的单词学会,你的代码也能写的很6,...【详细内容】
2022-10-14  风云教育  今日头条  Tags:Java   点击:(12)  评论:(0)  加入收藏
JVM 内存有以下区域: Young Generation Old Generation Metaspace Others region 图:JVM 内存区域 要查看哪些对象存储在哪个区域,您可以参考此视频剪辑。有时您的应用程序...【详细内容】
2022-10-04  qaseven   今日头条  Tags:Java   点击:(19)  评论:(0)  加入收藏
属性文件乱码通常是中文乱码,在英文下通常没有这个问题。 如上图显示的,中文字符在属性文件中读取后会显示为乱码.问题和解决导致这个问题的主要原因是属性文件如果你不进行设...【详细内容】
2022-10-04  松鼠工厂  今日头条  Tags:Java   点击:(14)  评论:(0)  加入收藏
JoinerGuava Joiner顾名思义就是将字符串连接起来 Joiner joiner = Joiner.on("; ").skipNulls(); System.out.println(joiner.join("Harry", null, "Ron", "Hermione"));...【详细内容】
2022-10-03  小盒子的技术分享    Tags:Java   点击:(26)  评论:(0)  加入收藏
站内最新
站内热门
站内头条