n*n矩阵,输出矩阵中任意两点之间所有路径

2024-08-28 17:04

本文主要是介绍n*n矩阵,输出矩阵中任意两点之间所有路径,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

题目1:给你一个正整数n, 构造一个n*n的四项链表矩阵。

要求: 1.使用四项链表

            2.矩阵从左到右,从上到下值依次为1,2,3,4,......n*n

题目2:基于题目1, 在n*n链表矩阵中,输出矩阵中任意两点之间所有路径。

要求: 1.不能使用全局变量

            2.方法只接收两个参数,分别为起始节点

            3.不能重复遍历

构造一个n*n的四项链表矩阵

思路: 要构建一个n*n的矩阵,有这个一个简单思路:

  1. 初始化i=0
  2. 先构建第i行,第i列链表
  3. 再构建(n-1)*(n-1)矩阵
  4. 将第二步链表与第三步矩阵连接起来
  5. i=i+1
  6. 重复第二步

/*** @param k 当前要构造K*K矩阵* @param n* @return*/
public Node<E> build(int k, int n) {if (k <= 0) {return null;}Node<E> head = new Node();// 右侧Node<E> r = head;int i = k - 1;while (i > 0) {Node<E> right = new Node();r.right = right;right.left = r;r = right;i--;}// 下面i = k - 1;Node<E> d = head;while (i > 0) {Node<E> down = new Node();d.down = down;down.up = d;d = down;i--;}// n-1矩阵  subHead n-1矩阵头节点Node<E> subHead = build(k - 1, n);// 先连接顶部Node<E> h = head.right;Node<E> subH = subHead;while (null != h) {h.down = subH;subH.up = h;h = h.right;subH = subH.right;}// 再连接左侧Node<E> down = head.down;Node<E> subDown = subHead;while (null != down) {down.right = subDown;subDown.left = down;down = down.down;subDown = subDown.down;}return head;
}

 输出矩阵中任意两点之间所有路径

思路: 深度优先搜索(DFS) + 回溯

  1. 如果起始节点相等,直接输出,结束
  2. 起两个栈, 一个主栈,一个副栈
    1. 主栈:存放线路上的节点,用于符合条件时直接输出
    2. 副栈:存放主栈节点的邻接节点集合,需考虑边界与重复遍历问题
  3. 起始节点入主栈,起始节点的邻接节点入副站
  4. 副栈弹出节点集合
    1. 弹出集合中第一个节点node(集合长度-1)
    2. 集合重新入副栈
  5. 如果node=null, 主栈、副栈执行pop()操作,表示需要回溯到上一步(可以理解为,当前无路可走,需要从上一个节点试图走其他方向), 跳转并重复第4步
  6. 如果node!=null, 将node节点压入主栈
  7. 此时如果node==终止节点,则输出主站所有节点,弹出主栈节点(此时==终止节点),跳转并重复第4步
  8. 将node邻接节点压入副栈

如下图5 * 5矩阵, 起始节点7 ,终止节点19

第一次                                                                       

第二次 

第三次

第四次                                     

第五次                                   

第六次

 第**次

  • 此时主栈栈顶元素为19,等于终止节点,则输出主占所有节点值
  • 弹出主栈栈顶节点19,从新取副栈栈顶节点
  • 当前主栈栈顶为18,试图从18开始再向其他方向寻找
  • 18邻接节点取出13,压住主栈

这样,不断压入弹出最终可遍历出所有路线

核心代码

public void search(Node<E> startNode, Node<E> endNode) {if (startNode == endNode) {System.out.println(startNode.v + " ");return;}// 主栈Stack<Node<E>> stack = new Stack<>();// 副栈Stack<Queue<Node<E>>> adjoinStack = new Stack<>();// 将开始节点入主栈stack.push(startNode);// 将起始节点邻接节点集合如副栈this.pushAdjoinStack(stack, startNode, adjoinStack);while (!stack.empty()) {Queue<Node<E>> queue = adjoinStack.pop();Node<E> node = queue.poll();adjoinStack.push(queue);// 如果到无路可走if (null == node) {adjoinStack.pop();stack.pop();continue;}stack.push(node);// 找到一个路径if (node == endNode) {this.printResult(stack);// 弹出主栈栈顶元素stack.pop();continue;}pushAdjoinStack(stack, node, adjoinStack);}
}/*** 将node节点的邻接几点入副站* 1. 边界问题* 2. 避免重复入副站** @param stack* @param node* @param adjoinStack*/
public void pushAdjoinStack(Stack<Node<E>> stack, Node<E> node, Stack<Queue<Node<E>>> adjoinStack) {Queue<Node<E>> nodeList = new LinkedList<>();if (node.up != null && !stack.contains(node.up)) {nodeList.offer(node.up);}if (node.down != null && !stack.contains(node.down)) {nodeList.offer(node.down);}if (node.left != null && !stack.contains(node.left)) {nodeList.offer(node.left);}if (node.right != null && !stack.contains(node.right)) {nodeList.offer(node.right);}adjoinStack.push(nodeList);
}

完整代码:

/*** 节点类** @author ywl* @version 1.0* @date 2024/8/21 20:01*/public class Node<E> {E v;Node<E> up;Node<E> down;Node<E> left;Node<E> right;// 省略 getter setter
}

import java.util.*;/*** 题目1:给你一个正整数n, 构造一个n*n的四项链表矩阵。* 题目2:基于题目1, 在n*n链表矩阵中,输出矩阵中任意两点之间所有路径。** @author ywl* @version 1.0* @date 2024/8/21 20:01*/
public class StLink<E> {public Node<E> build(int k, int n) {if (k <= 0) {return null;}Node<E> head = new Node();// 右侧Node<E> r = head;int i = k - 1;while (i > 0) {Node<E> right = new Node();r.right = right;right.left = r;r = right;i--;}// 下面i = k - 1;Node<E> d = head;while (i > 0) {Node<E> down = new Node();d.down = down;down.up = d;d = down;i--;}// n-1矩阵  subHead n-1矩阵头节点Node<E> subHead = build(k - 1, n);// 先连接顶部Node<E> h = head.right;Node<E> subH = subHead;while (null != h) {h.down = subH;subH.up = h;h = h.right;subH = subH.right;}// 再连接左侧Node<E> down = head.down;Node<E> subDown = subHead;while (null != down) {down.right = subDown;subDown.left = down;down = down.down;subDown = subDown.down;}return head;}public static void main(String[] args) {int n = 4;StLink<Integer> sl = new StLink<>();Node<Integer> head = sl.build(n, n);// 循环赋值Node<Integer> down = head;int c = 1;while (null != down) {Node<Integer> right = down;while (null != right) {right.setV(c++);right = right.right;}down = down.down;}/*** 1  2  3  4* 5  6  7  8* 9  10 11 12* 13 14 15 16*/sl.search(head.right, head.right.right.down.down);}public void search(Node<E> startNode, Node<E> endNode) {if (startNode == endNode) {System.out.println(startNode.v + " ");return;}// 主栈Stack<Node<E>> stack = new Stack<>();// 副栈Stack<Queue<Node<E>>> adjoinStack = new Stack<>();// 将开始节点入主栈stack.push(startNode);// 将起始节点邻接节点集合如副栈this.pushAdjoinStack(stack, startNode, adjoinStack);int maxLen = Integer.MAX_VALUE;List<List<E>> result = new ArrayList<>();while (!stack.empty()) {Queue<Node<E>> queue = adjoinStack.pop();Node<E> node = queue.poll();adjoinStack.push(queue);// 如果无路可走if (null == node) {adjoinStack.pop();stack.pop();continue;}stack.push(node);// 找到一个路径if (node == endNode) {List<E> es = this.printResult(stack);if(es.size() == maxLen) {result.add(es);} else if(es.size() < maxLen) {maxLen = es.size();result = new ArrayList<>();result.add(es);}// 弹出主栈栈顶元素stack.pop();continue;}pushAdjoinStack(stack, node, adjoinStack);}System.out.println("最小路径:");for(List<E> r : result) {for(int i = 0; i < r.size(); i++) {System.out.print(r.get(i)+" ");}System.out.println();}}/*** 将node节点的邻接几点入副站* 1. 边界问题* 2. 避免重复入副站** @param stack* @param node* @param adjoinStack*/public void pushAdjoinStack(Stack<Node<E>> stack, Node<E> node, Stack<Queue<Node<E>>> adjoinStack) {Queue<Node<E>> nodeList = new LinkedList<>();if (node.up != null && !stack.contains(node.up)) {nodeList.offer(node.up);}if (node.down != null && !stack.contains(node.down)) {nodeList.offer(node.down);}if (node.left != null && !stack.contains(node.left)) {nodeList.offer(node.left);}if (node.right != null && !stack.contains(node.right)) {nodeList.offer(node.right);}adjoinStack.push(nodeList);}private List<E> printResult(Stack<Node<E>> stack) {List<E> r = new ArrayList<>();Iterator<Node<E>> iterator = stack.iterator();while (iterator.hasNext()) {Node<E> next = iterator.next();r.add(next.v);System.out.print(next.v + " ");}System.out.println();return r;}}

这篇关于n*n矩阵,输出矩阵中任意两点之间所有路径的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1115358

相关文章

Python一次性将指定版本所有包上传PyPI镜像解决方案

《Python一次性将指定版本所有包上传PyPI镜像解决方案》本文主要介绍了一个安全、完整、可离线部署的解决方案,用于一次性准备指定Python版本的所有包,然后导出到内网环境,感兴趣的小伙伴可以跟随... 目录为什么需要这个方案完整解决方案1. 项目目录结构2. 创建智能下载脚本3. 创建包清单生成脚本4

python获取指定名字的程序的文件路径的两种方法

《python获取指定名字的程序的文件路径的两种方法》本文主要介绍了python获取指定名字的程序的文件路径的两种方法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要... 最近在做项目,需要用到给定一个程序名字就可以自动获取到这个程序在Windows系统下的绝对路径,以下

mybatis-plus如何根据任意字段saveOrUpdateBatch

《mybatis-plus如何根据任意字段saveOrUpdateBatch》MyBatisPlussaveOrUpdateBatch默认按主键判断操作类型,若需按其他唯一字段(如agentId、pe... 目录使用场景方法源码方法改造首先在service层定义接口service层接口实现总结使用场景my

SpringBoot路径映射配置的实现步骤

《SpringBoot路径映射配置的实现步骤》本文介绍了如何在SpringBoot项目中配置路径映射,使得除static目录外的资源可被访问,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一... 目录SpringBoot路径映射补:springboot 配置虚拟路径映射 @RequestMapp

从基础到高级详解Python数值格式化输出的完全指南

《从基础到高级详解Python数值格式化输出的完全指南》在数据分析、金融计算和科学报告领域,数值格式化是提升可读性和专业性的关键技术,本文将深入解析Python中数值格式化输出的相关方法,感兴趣的小伙... 目录引言:数值格式化的核心价值一、基础格式化方法1.1 三种核心格式化方式对比1.2 基础格式化示例

java -jar example.jar 产生的日志输出到指定文件的方法

《java-jarexample.jar产生的日志输出到指定文件的方法》这篇文章给大家介绍java-jarexample.jar产生的日志输出到指定文件的方法,本文给大家介绍的非常详细,对大家的... 目录怎么让 Java -jar example.jar 产生的日志输出到指定文件一、方法1:使用重定向1、

Java中数组与栈和堆之间的关系说明

《Java中数组与栈和堆之间的关系说明》文章讲解了Java数组的初始化方式、内存存储机制、引用传递特性及遍历、排序、拷贝技巧,强调引用数据类型方法调用时形参可能修改实参,但需注意引用指向单一对象的特性... 目录Java中数组与栈和堆的关系遍历数组接下来是一些编程小技巧总结Java中数组与栈和堆的关系关于

在Java中实现线程之间的数据共享的几种方式总结

《在Java中实现线程之间的数据共享的几种方式总结》在Java中实现线程间数据共享是并发编程的核心需求,但需要谨慎处理同步问题以避免竞态条件,本文通过代码示例给大家介绍了几种主要实现方式及其最佳实践,... 目录1. 共享变量与同步机制2. 轻量级通信机制3. 线程安全容器4. 线程局部变量(ThreadL

Spring Boot集成/输出/日志级别控制/持久化开发实践

《SpringBoot集成/输出/日志级别控制/持久化开发实践》SpringBoot默认集成Logback,支持灵活日志级别配置(INFO/DEBUG等),输出包含时间戳、级别、类名等信息,并可通过... 目录一、日志概述1.1、Spring Boot日志简介1.2、日志框架与默认配置1.3、日志的核心作用

python设置环境变量路径实现过程

《python设置环境变量路径实现过程》本文介绍设置Python路径的多种方法:临时设置(Windows用`set`,Linux/macOS用`export`)、永久设置(系统属性或shell配置文件... 目录设置python路径的方法临时设置环境变量(适用于当前会话)永久设置环境变量(Windows系统