首页 > 其他分享 >寻找两个链表相交节点方法(可以是有环链表)

寻找两个链表相交节点方法(可以是有环链表)

时间:2023-11-02 17:25:39浏览次数:39  
标签:Node cur2 cur1 next 链表 环链 null 节点

问题分析:两个链表相交可以分为两个大类,一是两个无环链表相交,二是两个有环链表相交。

 

无环相交如图:

有环相交有两种情况,一种是 先相交后成环,如图:

另一种是交点有两个,是成环后的交点(入环节点不同)

 

方法

1.判断链表是否有环,返回第一个入环节点。 2.判断是否相交 3.判断相交节点是否相同

判断链表是否有环,并返回第一个入环节点

使用快慢指针,快指针一次走两步,慢指针一次走一步,如果链表有环则两个指针必然会相遇。具体细节参考判断链表是否有环

无环链表相交问题

1.判断两个链表是否同时无环 2.先遍历两个链表获得链长lensA和lensB 3.让长链表先走abs(lensA - lensB)步,之后两个链表共同前进,如果在遍历链表结束之前有相同节点,则两个链表相交。

两个有环链表相交问题

1.判断两个链表是否同时有环 2.判断链表第一个入环节点是否相同。 3.如果相同,则使用无环链表相交问题类似的方法,只是将链表遍历终点定在入环节点 3.如果不相同,则从链表A开始遍历,如果遍历过程中遇见了链表B的入环节点则链表相交。 代码如下:

public class FindFirstIntersectNode {
//定义节点
	public static class Node {
		public int value;
		public Node next;

		public Node(int data) {
			this.value = data;
		}
	}

	public static Node getIntersectNode(Node head1, Node head2) {
		if (head1 == null || head2 == null) {
			return null;
		}
		//判断链表是否有环,有就返回环节点
		Node loop1 = getLoopNode(head1);
		Node loop2 = getLoopNode(head2);
		//都无环,采用无环相交方法
		if (loop1 == null && loop2 == null) {
			return noLoop(head1, head2);
		}
		//都有环,采用有环相交方法
		if (loop1 != null && loop2 != null) {
			return bothLoop(head1, loop1, head2, loop2);
		}
		return null;
	}

	// 快慢指针方法,找到链表第一个入环节点,如果无环,返回null
	public static Node getLoopNode(Node head) {
		if (head == null || head.next == null || head.next.next == null) {
			return null;
		}
		// n1 慢  n2 快
		Node n1 = head.next; // n1 -> slow
		Node n2 = head.next.next; // n2 -> fast
		while (n1 != n2) {
			if (n2.next == null || n2.next.next == null) {
				return null;
			}
			n2 = n2.next.next;
			n1 = n1.next;
		}
		//相遇时,快指针返回头结点,每次走一步
		n2 = head; // 
		while (n1 != n2) {
			n1 = n1.next;
			n2 = n2.next;
		}
		return n1;
	}

	// 如果两个链表都无环,返回第一个相交节点,如果不想交,返回null
	public static Node noLoop(Node head1, Node head2) {
		if (head1 == null || head2 == null) {
			return null;
		}
		Node cur1 = head1;
		Node cur2 = head2;
		int n = 0;
		while (cur1.next != null) {
			n++;
			cur1 = cur1.next;
		}
		while (cur2.next != null) {
			n--;
			cur2 = cur2.next;
		}
		if (cur1 != cur2) {
			return null;
		}
		// n  :  链表1长度减去链表2长度的值
		cur1 = n > 0 ? head1 : head2; // 谁长,谁的头变成cur1
		cur2 = cur1 == head1 ? head2 : head1; // 谁短,谁的头变成cur2
		n = Math.abs(n);
		while (n != 0) {
			n--;
			cur1 = cur1.next;
		}
		while (cur1 != cur2) {
			cur1 = cur1.next;
			cur2 = cur2.next;
		}
		return cur1;
	}

	// 两个有环链表,返回第一个相交节点,如果不想交返回null
	public static Node bothLoop(Node head1, Node loop1, Node head2, Node loop2) {
		Node cur1 = null;
		Node cur2 = null;
		//入环节点相同
		if (loop1 == loop2) {
			cur1 = head1;
			cur2 = head2;
			int n = 0;
			while (cur1 != loop1) {
				n++;
				cur1 = cur1.next;
			}
			while (cur2 != loop2) {
				n--;
				cur2 = cur2.next;
			}
			cur1 = n > 0 ? head1 : head2;
			cur2 = cur1 == head1 ? head2 : head1;
			n = Math.abs(n);
			while (n != 0) {
				n--;
				cur1 = cur1.next;
			}
			while (cur1 != cur2) {
				cur1 = cur1.next;
				cur2 = cur2.next;
			}
			return cur1;
		} else {//入环节点不同
			cur1 = loop1.next;
			while (cur1 != loop1) {
				if (cur1 == loop2) {
					return loop1;
				}
				cur1 = cur1.next;
			}
			return null;
		}
	}
}

标签:Node,cur2,cur1,next,链表,环链,null,节点
From: https://www.cnblogs.com/soliang/p/17805839.html

相关文章

  • 前端歌谣的刷题之路-第六十八题-js动态创建节点
     前言我是歌谣我有个兄弟巅峰的时候排名c站总榜19叫前端小歌谣曾经我花了三年的时间创作了他现在我要用五年的时间超越他今天又是接近兄弟的一天人生难免坎坷大不了从头再来歌谣的意志是永恒的放弃很容易但是坚持一定很酷本题目源自于牛客网微信公众号前端小歌谣题目......
  • 双向链表结构分析
    双向链表描述双向链表也叫双链表,它的每个数据结点都有两个指针,分别指向前驱结点和后继节点,同时有一个数据域来保存数据,双向链表的图示如下:从图片可以看出,双链表的头结点的前驱结点和尾结点的后继结点为空,这一点要注意,对双链表的操作要检查这两种情况。双向链表结构每个数据结......
  • 验证2个节点udp和tcp可通性
    -u表示udp,默认是tcp。-l表示作为server监听。server:192.168.0.104上开启udp123端口server发送11client:连接192.168.0.104上udp123端口client发送100 server:192.168.0.104上开启tcp123端口server发送102client:连接192.168.0.104上tcp123端口client发送101......
  • 牛客剑指offer刷题链表篇
    @[TOC]从尾到头打印链表题目输入一个链表的头节点,从尾到头反过来打印出每个节点的值。思路利用栈后进先出的性质实现;代码实现privatestaticvoidprintReverseNode(ListNodehead){if(head==null){return;}ListNodepNode=head;......
  • 《安富莱嵌入式周报》第320期:键盘敲击声解码, 军工级boot设计,开源CNC运动控制器,C语言
     视频版:https://www.bilibili.com/video/BV1Cr4y1d7Mp/1、键盘敲击声解码https://arxiv.org/abs/2308.01074键盘敲击声被解码的话,我们使用键盘输入密码将被方便的解码出来。这篇文章介绍了一种使用最先进的深度学习模型,以便使用手机麦克风对笔记本电脑敲击键盘分析。实际测试训练......
  • dolphinscheduler worker节点挂掉后连接不上
    Errorcreatingbeanwithname'zookeeperRegistry':Invocationofinitmethodfailed;nestedexceptionisorg.apache.dolphinscheduler.registry.api.RegistryException:zookeeperconnecttimeout在worker日志中可以发现报这个错误。注册不上zk,连接超时。这是因......
  • 倾斜摄影三维模型的根节点合并注意事项浅析
    倾斜摄影三维模型的根节点合并注意事项浅析 倾斜摄影三维模型的根节点合并是构建高精度、真实感和稳定性的三维模型的关键步骤之一。在进行根节点合并时,需要注意以下几个重要的事项,以确保合并的准确性和可靠性。首先,准确的相机标定是进行根节点合并的基础。相机标定是确定相......
  • 两种方式讲链表节点删除
    第一种讲法就是循环的方式,因为要循环遍历这个链表,所以我们会运用到一个很重要的哨兵思想,就是定一个没啥意义的哨兵,让head“makesense”,接着,我们的任务是对链表进行删除,那就涉及到一个前端链表的指向问题,但是现在这个是单向链表,我们无法知道你前面那个是谁,所以我们也可以想办法......
  • 141. 环形链表
    目录题目法一、快慢指针法二、哈希表题目给你一个链表的头节点head,判断链表中是否有环。如果链表中有某个节点,可以通过连续跟踪next指针再次到达,则链表中存在环。为了表示给定链表中的环,评测系统内部使用整数pos来表示链表尾连接到链表中的位置(索引从0开始)。注意:po......
  • 234.回文链表
    目录题目法一、复制+反转链表法二、堆栈法三、快慢指针和链表反转题目给你一个单链表的头节点head,请你判断该链表是否为回文链表。如果是,返回true;否则,返回false示例1:输入:head=[1,2,2,1]输出:true示例2:输入:head=[1,2]输出:false法一、复制+反转链表把原......