Java 递归 跳出死循环

article/2025/9/13 15:11:08

Java 递归 跳出死循环

学习了知道树形数据,如何遍历子集,生成List数据,如果数据里面是带各种循环的呢? 要如何处理? 如何跳出死循环?

要求:

根据当前环节id,比如18,找到前置路径: 17到11; 或是14的前置路径: 13,12,11

 

数据是树形变成后的List数据:

[{"TCH_MOD": 11,"NEXT_TCH_MOD": 12
}, {"TCH_MOD": 12,"NEXT_TCH_MOD": 13
}, {"TCH_MOD": 13,"NEXT_TCH_MOD": 14
}, {"TCH_MOD": 14,"NEXT_TCH_MOD": 15
}, {"TCH_MOD": 15,"NEXT_TCH_MOD": 16
}, {"TCH_MOD": 16,"NEXT_TCH_MOD": 17
}, {"TCH_MOD": 17,"NEXT_TCH_MOD": 18
}, {"TCH_MOD": 12,"NEXT_TCH_MOD": 14
}, {"TCH_MOD": 13,"NEXT_TCH_MOD": 15
}, {"TCH_MOD": 15,"NEXT_TCH_MOD": 17
}]

思路:

    在递归的时候,对数据进行标记,出现了多次的,就不进行处理。这样就不会进行死循环中。 (PS:递归的使用,除了找到规律,判断递归终止条件也是很重要的。有时候找不到终止条件,主要是对整个过程不理解,或是一知半解)

处理:

代码:

import org.apache.commons.collections4.ListUtils;
import org.apache.commons.collections4.MapUtils;
import org.apache.commons.lang3.StringUtils;import java.util.*;
import java.util.stream.Collectors;public class RemarkRecurTreeList {private List<Map<String, Object>> treeNodeList;private String rootValue;// 添加一个list作为标记用于判断是否循环private List<String> preValues = new ArrayList<>();public RemarkRecurTreeList(List<Map<String, Object>> treeNodeList, String rootValue) {this.treeNodeList = treeNodeList;this.rootValue = rootValue;}//建立树形结构public List<Map<String, Object>> builTree(String parentTag, String childTag) {List<Map<String, Object>> rootNodeList = getRootNode(parentTag);LinkedList<Map<String, Object>> flatTreeList = new LinkedList<>();preValues.add(rootValue);for (Map<String, Object> e : ListUtils.emptyIfNull(rootNodeList)) {buildChilTree(e, parentTag, childTag, flatTreeList);flatTreeList.add(e);}return ListUtils.emptyIfNull(flatTreeList).stream().filter(Objects::nonNull).collect(Collectors.toList());}//递归,建立子树形结构  返回private Map<String, Object> buildChilTree(Map<String, Object> pNode, String parentTag, String childTag, List<Map<String, Object>> flatTreeList) {String nextValue1 = MapUtils.getString(pNode, childTag);if(!preValues.contains(nextValue1)){preValues.add(nextValue1); // 添加标记for (Map<String, Object> e : ListUtils.emptyIfNull(treeNodeList)) {// 跟上一个进行匹配String curValue = MapUtils.getString(e, parentTag);String nextValue = MapUtils.getString(pNode, childTag);if (StringUtils.equals(curValue, nextValue) ) { // && !curValue.equals(rootValue)flatTreeList.add(buildChilTree(e, parentTag, childTag, flatTreeList));}}return pNode;}return null;}//获取根节点private List<Map<String, Object>> getRootNode(String parentTag) {List<Map<String, Object>> rootNodeLists = new ArrayList<>();for (Map<String, Object> e : ListUtils.emptyIfNull(treeNodeList)) {if (StringUtils.equals(MapUtils.getString(e, parentTag), rootValue)) {rootNodeLists.add(e);}}return rootNodeLists;}}

测试:

import com.alibaba.fastjson.JSONArray;
import com.alibaba.fastjson.JSONObject;
import org.apache.commons.collections4.ListUtils;import java.util.List;
import java.util.Map;
import java.util.stream.Collectors;public class RemarkRecurTreeListTest {public static void main(String[] args){List<Map<String,Object>> areaList = changeFormat(initData());System.out.println("init data "+ areaList.toString());// 构建树形RemarkRecurTreeList commonTree = new RemarkRecurTreeList(areaList, "18");List<Map<String, Object>> result = commonTree.builTree("NEXT_TCH_MOD", "TCH_MOD");System.out.println();result.forEach(e -> {System.out.println(e.get("TCH_MOD")+ " <-- "+ e.get("NEXT_TCH_MOD"));});}private static List<Map<String,Object>> changeFormat(String areaInfo){JSONArray areaArr = JSONArray.parseArray(areaInfo);return ListUtils.emptyIfNull(areaArr).stream().map(e -> (JSONObject) e).map(e -> (Map<String, Object>)JSONObject.parseObject( e.toJSONString())).collect(Collectors.toList());}private static String initData(){// 初始循环的return "[{\"TCH_MOD\":11,\"NEXT_TCH_MOD\":12},{\"TCH_MOD\":12,\"NEXT_TCH_MOD\":13},{\"TCH_MOD\":13,\"NEXT_TCH_MOD\":14},{\"TCH_MOD\":14,\"NEXT_TCH_MOD\":15},{\"TCH_MOD\":15,\"NEXT_TCH_MOD\":16},{\"TCH_MOD\":16,\"NEXT_TCH_MOD\":17},{\"TCH_MOD\":17,\"NEXT_TCH_MOD\":18},{\"TCH_MOD\":12,\"NEXT_TCH_MOD\":14},{\"TCH_MOD\":13,\"NEXT_TCH_MOD\":15},{\"TCH_MOD\":15,\"NEXT_TCH_MOD\":17}]";}}

结果:

init data [{"NEXT_TCH_MOD":12,"TCH_MOD":11}, {"NEXT_TCH_MOD":13,"TCH_MOD":12}, {"NEXT_TCH_MOD":14,"TCH_MOD":13}, {"NEXT_TCH_MOD":15,"TCH_MOD":14}, {"NEXT_TCH_MOD":16,"TCH_MOD":15}, {"NEXT_TCH_MOD":17,"TCH_MOD":16}, {"NEXT_TCH_MOD":18,"TCH_MOD":17}, {"NEXT_TCH_MOD":14,"TCH_MOD":12}, {"NEXT_TCH_MOD":15,"TCH_MOD":13}, {"NEXT_TCH_MOD":17,"TCH_MOD":15}]11 <-- 12
12 <-- 13
13 <-- 14
14 <-- 15
15 <-- 16
16 <-- 17
17 <-- 18

总结:

   在递归有循环指向的数据的时候,要进行标记处理,找到跳出循环的条件,不然会出现递归死循环的情况。


http://chatgpt.dhexx.cn/article/cDoja8VV.shtml

相关文章

java递归实现汉诺塔

汉诺塔的规则是:一共三根柱子,一根柱子从上到下套着有小到大的若干个圆盘,要将所有圆盘按照这个排放顺序移动到第三根柱子上,并且每次只能移动一个圆盘. 可以将整个过程分为三个步骤来看: 第一步:将除最大圆盘外的n-1个圆盘移动辅助柱子上 第二步:将最大的圆盘移动到目标柱子 …

Java递归生成树

1.建菜单表 CREATE TABLE t_menu (id int(11) NOT NULL AUTO_INCREMENT,pid int(11) NOT NULL,name varchar(255) DEFAULT NULL,PRIMARY KEY (id) ) ENGINEInnoDB AUTO_INCREMENT11 DEFAULT CHARSETutf8mb4;2.造一些数据 注意&#xff1a;根节点的pid0&#xff0c;其他节点的p…

Java递归练习

目录 1. 递归求N的阶乘 2. 按顺序打印一个数字的每一位 3. 递归求n的和 4. 计算一个数每一位之和 5. 求斐波那契数列的第N项 6. 汉诺塔 7. 青蛙跳台阶 1. 递归求N的阶乘 //递归求n的阶乘 public class test {//求n的阶乘方法public static int func(int n) {if (n 1) …

java递归查询

递归建立数据库的字段要建立pid 既pid是父id 这里的pid要能在id找到 找到就说明当前id的组织是父组织 如果找不到就说明的顶级id 图中的主菜单1就是顶级组织 public Menu getMenuTree() {//先查询顶级目录LambdaQueryWrapper<Menu> menuLambdaQueryWrapper new LambdaQ…

java 积累递归深度_java递归的深度

递归的深度 在使用递归的时候经常会抛出StackOverflowError&#xff0c;顾名思义就是栈满了&#xff0c;而我们这里所说的栈在java中通常就是虚拟机栈(vm stack)&#xff0c;在每个方法执行的同时都会创建一个栈帧&#xff0c;用于存储局部变量表、操作数栈&#xff0c;动态链接…

Java递归全排列

Java递归全排列 目录 1&#xff0c; 全排列实现思路 排列组合概念 理解 2&#xff0c;全排列的重点 3&#xff0c;全排列java实现 Java: 结果&#xff1a; 总结&#xff1a; 1&#xff0c; 全排列实现思路 排列组合概念 排列&#xff0c;就是指从给定n个数的元素中取出…

Java递归优化

目录 一、经典递归二、尾递归优化 一、经典递归 &#xff08;注&#xff1a;本文例子只用于探讨&#xff0c;不考虑n<0 等复杂情况。&#xff09; int factorial(int n){if(n1){return 1;}else{return n*factorial(n-1);} }执行过程如下&#xff1a; factorial&#xff0…

java中的方法递归

JAVA中的方法递归 递归的思路代码举例 一、递归的思路 一个方法在执行时&#xff0c;调用自身被称为“递归”。 递归相当于数学归纳法&#xff0c;有一个起始条件&#xff0c;有一个递推公式。 递归可以分为&#xff1a;单路递归和多路递归&#xff08;如二叉树和斐波那契数…

Java递归简介

Java递归简介 Java中的递归定义为“方法直接或间接调用自身&#xff08;相同方法&#xff09;”。递归函数用于需要一次又一次地执行同一组操作直到达到结果的情况。它执行几次迭代&#xff0c;并且每次迭代问题说明都变得越来越简单。Java中的递归是一种基于对同一问题的较小…

Java中的递归详解

文章目录 概述递归累加求和计算1 ~ n的和代码执行图解 递归求阶乘递归打印多级目录综合案例文件搜索 文件过滤器优化Lambda优化 概述 递归&#xff1a;指在当前方法内调用自己的这种现象。递归的分类: 递归分为两种&#xff0c;直接递归和间接递归。直接递归称为方法自身调用自…

fstream的操作

第一段转载自&#xff1a;https://blog.csdn.net/jaster_wisdom/article/details/52400059 在C中输入输出到指定文件&#xff0c;或者从指定文件中读出数据使用fstream类较为方便。 1.将数据写到磁盘的指定文件中 首先第一步是加头文件#include <fstream>,引入库函数 第二…

fstream知识梳理

fstream知识梳理 fstream头文件中的3中类型fstream关联的文件方式读写文件fstream的操作函数eof()函数bad()函数good()函数clear()函数get() 和 put()getline()函数 fstream头文件中的3中类型 ifstream :从一个文件中读取数据ofstream :从一个文件中写入数据fstream :既可以读…

linux fstream open,fstream创建文件的问题

在C语言中&#xff0c;使用FILE*对文件进行操作。当文件不存在时&#xff0c;如果采用w模式打开&#xff0c;文件会自动创建&#xff0c;第二个参数可以为&#xff1a; 如上图所示&#xff0c;只有当以w,w,a,a打开时&#xff0c;文件才会自动创建&#xff0c;而 r,r则不能。 那…

fstream

ifstream -- 从磁盘中写入内存 ofstream -- 从内存中输出到磁盘 #include <fstream> ofstream //文件写操作,内存写入存储设备(文件) 输出流 ifstream //文件读操作,存储设备到内存. 输入流 fstream //读写操作,对打开的文件可进行读写. 前两者的…

fstream读写文件

1.fstream是什么&#xff1f; fstream是C标准库中面向对象库的一个&#xff0c;用于操作流式文件。 fstream本质上是一个class&#xff0c;提供file操作的各种方法。 2.关系图 basic_fstream是一个类模板&#xff0c;暂且不用深入理解它。我们关心的事&#xff0c;它前面继承…

【C++入门】文件流(fstream)介绍和使用

1、打开函数&#xff1a;open mode含义ios::in以读取方式打开文件ios::out以写入方式打开文件ios::binary以二进制方式存取ios::ate存取指针在文件末尾ios::app写入时采用追加方式ios::trunc写入时抹去旧数据 (1)在C11标准时&#xff0c;open函数的文件路径可以传char指针也可以…

谷歌浏览器代理服务器出现问题怎么办?(最快的解决办法)

谷歌浏览器代理服务器出现问题怎么办? 两个解决办法&#xff08;其他浏览器也可用&#xff09;&#xff1a; &#xff08;1&#xff09;方法一&#xff1a;【重置】浏览器。&#xff08;最方便快捷的方式&#xff09; ①进入“设置”——&#xff09;选择“重置并清理”——…

上不了网,代理服务器出现问题

上不了网&#xff0c;代理服务器出现问题 1.问题截图 2.解决的方法

代理服务器可能有问题,或地址不正确的解决方案

可能是设置了代理服务器的原因。 1.先打开代理服务器设置 2.关闭代理服务器。

win10 代理服务器出现问题 或者地址有误

本人win10系统&#xff0c;今天晚上使用浏览器&#xff0c;打开页面的时候突然出现了&#xff1a; 代理服务器出现问题 或者地址有误。 具体诊断&#xff0c;错误截图也就不放了。 解决办法&#xff1a;原本那个使用代理服务器不知道怎么的打开了&#xff0c;关掉之后再通过…