00.01.2021-5-27.java实现检查有向图是否出现闭环

以下代码声明了一个节点类,分别根据parentId和childId来判断新增父子关系是否导致有闭环的方法。

  • 产生需求的原因是执行流程不允许出现循环。
package com.test.demo.tu;

import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;

import java.util.*;

public class CheckLoop {
    @Data
    @AllArgsConstructor
    @NoArgsConstructor
    public static class Node {//节点类
        private Integer id;
        private List<Integer> parentIds;//表示上一个节点的id集合
        private List<Integer> childIds;//表示下一个节点的id集合
    }

    public static void main(String[] args) {
        //假设现在要链接一根从5指向3的箭头
        long l = System.currentTimeMillis();
        Integer preId = 5, nextId = 3;
        //创建测试数据
        Map<Integer, Node> map = create();
        //存储已经检查合格的过的节点
        Set<Integer> checked = new HashSet<>();
        //存储已经走过的节点
        Set<Integer> repeat = new HashSet<>();
        //判断当前节点和下一个节点只要有一个不在就失败
        if (map.get(nextId) == null||map.get(preId)==null) {
            return;
        }
        //检查是否有闭环
        boolean result = execute(preId, nextId, map, checked, repeat);
        System.out.println(checked);
        System.out.println(repeat);
        System.out.println(result);
        System.out.println(System.currentTimeMillis() - l);

        //假设现在要链接一根从5指向3的箭头
        long l1 = System.currentTimeMillis();
        Integer preId1 = 5, nextId1 = 3;
        //创建测试数据
        Map<Integer, Node> map1 = create();
        //存储已经检查合格的过的节点
        Set<Integer> checked1 = new HashSet<>();
        //存储已经走过的节点
        Set<Integer> repeat1 = new HashSet<>();
        //判断当前节点和下一个节点只要有一个不在就失败
        if (map1.get(nextId1) == null||map1.get(preId1)==null) {
            return;
        }
        //检查是否有闭环
        boolean result1 = execute1(nextId1,preId1 , map1, checked1, repeat1);
        System.out.println(checked1);
        System.out.println(repeat1);
        System.out.println(result1);
        System.out.println(System.currentTimeMillis() - l1);
    }

    //通过父节点进行判断
    public static boolean execute(int currId, int nextId, Map<Integer, Node> map, Set<Integer> checked, Set<Integer> repeat) {
        //如果当前节点是已经检查通过的节点则直接返回检查成功
        if (checked.contains(currId)) {
            return true;
        }
        //如果当前节点不在检查通过的节点中但是又重复出现了,则直接返回重复
        if (repeat.contains(currId)) {
            return false;
        }
        //当前节点不在以上两个list中则将当前节点存入
        repeat.add(currId);
        //取出当前节点
        Node node = map.get(currId);
        List<Integer> parentIds = node.getParentIds();
        //判断如果没有父节点则直接返回校验成功
        if (parentIds==null||parentIds.size() == 0 ){
            return true;
        }
        //判断父id是否包含目标节点如果有则直接校验失败
        if (parentIds.contains(nextId)) {
            return false;
        } else {
            for (Integer parentId : parentIds) {
                //循环处理父id如果有校验失败的直接返回,否则将此父id放入校验合格list中
                if (!execute(parentId, nextId, map, checked, repeat)) {
                    return false;
                } else {
                    checked.add(parentId);
                }
            }
        }
        //走到这证明没在前面返回,则这里把本节点放入检查成功的列表中
        checked.add(currId);
        return true;
    }


    //通过子节点进行判断
    public static boolean execute1(int currId, int preId, Map<Integer, Node> map, Set<Integer> checked, Set<Integer> repeat) {
        //如果当前节点是已经检查通过的节点则直接返回检查成功
        if (checked.contains(currId)) {
            return true;
        }
        //如果当前节点不在检查通过的节点中但是又重复出现了,则直接返回重复
        if (repeat.contains(currId)) {
            return false;
        }
        //当前节点不在以上两个list中则将当前节点存入
        repeat.add(currId);
        //取出当前节点
        Node node = map.get(currId);
        List<Integer> childIds = node.getChildIds();
        //判断如果没有子节点则直接返回校验成功
        if (childIds!=null&&childIds.size() > 0 ){
            //判断子id是否包含目标节点如果有则直接校验失败
            if (childIds.contains(preId)) {
                return false;
            } else {
                for (Integer childId : childIds) {
                    //循环处理子id如果有校验失败的直接返回,否则将此父id放入校验合格list中
                    if (!execute1(childId, preId, map, checked, repeat)) {
                        return false;
                    } else {
                        checked.add(childId);
                    }
                }
            }
        }
        //走到这证明没在前面返回,则这里把本节点放入检查成功的列表中
        checked.add(currId);
        return true;
    }

    //创建测试数据
    public static Map<Integer, Node> create() {
        Node node0 = new Node();
        node0.setId(1);
        List<Integer> list0 = new ArrayList<>();
        list0.add(1);
        node0.setChildIds(list0);
        node0.setParentIds(new ArrayList<>());

        Node node1 = new Node();
        node1.setId(1);
        List<Integer> list1 = new ArrayList<>();
        list1.add(2);
        list1.add(3);
        node1.setChildIds(list1);
        List<Integer> parent1 = new ArrayList<>();
        parent1.add(0);
        node1.setParentIds(parent1);

        Node node2 = new Node();
        node2.setId(2);
        List<Integer> list2 = new ArrayList<>();
        list2.add(4);
        node2.setChildIds(list2);
        List<Integer> parent2 = new ArrayList<>();
        parent2.add(1);
        node2.setParentIds(parent2);

        Node node3 = new Node();
        node3.setId(3);
        List<Integer> list3 = new ArrayList<>();
        list3.add(4);
        node3.setChildIds(list3);
        List<Integer> parent3 = new ArrayList<>();
//        parent3.add(5);
        parent3.add(1);
        node3.setParentIds(parent3);

        Node node4 = new Node();
        node4.setId(4);
        List<Integer> list4 = new ArrayList<>();
        list4.add(5);
        node4.setChildIds(list4);
        List<Integer> parent4 = new ArrayList<>();
        parent4.add(3);
        parent4.add(2);
        node4.setParentIds(parent4);


        Node node5 = new Node();
        node5.setId(5);
        List<Integer> list5 = new ArrayList<>();
        list5.add(3);
        node5.setChildIds(list5);
        List<Integer> parent5 = new ArrayList<>();
        parent5.add(4);
        node5.setParentIds(parent5);
        Node node6 = new Node();
        node6.setId(6);
        Map<Integer, Node> map = new HashMap<>();
        map.put(0, node0);
        map.put(1, node1);
        map.put(2, node2);
        map.put(3, node3);
        map.put(4, node4);
        map.put(5, node5);
        map.put(6, node6);
        System.out.println(map);
        return map;
    }
}

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值