[牛客网刷题 Day2] JZ52 两个链表的第一个公共结点(没做出来)双指针巧解

本文主要是介绍[牛客网刷题 Day2] JZ52 两个链表的第一个公共结点(没做出来)双指针巧解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

题目描述:

在这里插入图片描述

思考过程:

好像想的太复杂了,首先固定phead1,寻找val一样的phead2,找到的话,就都往右移动一位;否则phead+1。可是需要注意好多好多的边界啊,写了好久好久,最后还是好几个用例通不过,只能根据用例慢慢改,可是怎么改都不对。o(╥﹏╥)o

class Solution:def FindFirstCommonNode(self , pHead1 , pHead2):# write code hereif pHead1==None or pHead2==None:return Nonedummy2 = pHead2while pHead1.next:while pHead1.val != pHead2.val:pHead2 = pHead2.next# 找完了都找不到相同值,l1+1,l2返回初始位置if pHead2 == None:pHead1 = pHead1.nextpHead2 = dummy2if pHead1 == None:return None# 找到相同值,都移动一位,dummy重新赋值else:dummy2 = pHead2if not(pHead1.next and pHead2.next):pHead2 = dummy2continuewhile pHead1.next or pHead2.next:if pHead1.next.val == pHead2.next.val:pHead1 = pHead1.nextpHead2 = pHead2.nextif pHead1.next == None and pHead2.next == None:return dummy2else:pHead1 = pHead1.nextif pHead1 == None:return pHead2.nextbreakelse: return dummy2return None

答案:

常规思路,跟我的思路差不多,但是他用到了哈希表。
啊啊啊,原来链表是可以比较的!可以使用in,可是我的编译器不能这样操作。。。
所以我是能做出这道题的呢!

啊啊啊啊啊啊,发现问题了!!我初始化链表的时候搞错了!应该创建三个链表,前两个再指向第三个呢!
在这里插入图片描述

常规思路就是用一个哈希集合来存储第一个链表遍历后的所有节点;接着遍历第二个链表,与哈希集合中的节点进行比较

如果第二个链表中节点不存在于哈希集合中,那么移动到下一个节点再比较
否则,节点存在于哈希集合中,返回第二个链表的该节点(他就是我们要找的第一个公共节点)。
PS:如果比较到最后一个节点发现,第二个链表中的节点都不存在于哈希集合中,那么说明这两个链表不相交!直接返回None即可。

# class ListNode:
#     def __init__(self, x):
#         self.val = x
#         self.next = None#
# 
# @param pHead1 ListNode类 
# @param pHead2 ListNode类 
# @return ListNode类
#第一种解法:哈希
class Solution:def FindFirstCommonNode(self , pHead1 , pHead2 ):# write code here#首先判断两个链表是否为空if pHead1 is None or pHead2 is None:return None#定义链表1 的集合set_A = set()node1, node2 = pHead1, pHead2 #定义两个节点#遍历链表 1 ,把每个节点加入集合中while node1:set_A.add(node1)node1 = node1.next#遍历链表2 看当前节点是否在 集合中;如果存在,当前节点就是要找的第一个公共节点;否则继续比较下一个节点。#这里还要注意,如果遍历完链表 B,发现所有节点都不在集合中,则说明两个链表不相交,返回None。while node2:if node2 in set_A:return node2node2 = node2.nextreturn None

还有一个复杂的双指针思路,啊!
哇!太巧妙了吧!原理是:走一遍pHead1,再走一遍pHead2的总路程一定是相等的!

# class ListNode:
#     def __init__(self, x):
#         self.val = x
#         self.next = None
# @param pHead1 ListNode类 
# @param pHead2 ListNode类 
# @return ListNode类
#第二种解法:双指针
class Solution:def FindFirstCommonNode(self , pHead1 , pHead2 ):# write code here#首先判断两个链表是否为空if pHead1 is None or pHead2 is None:return None#使用双指针思路进行判断p1, p2 = pHead1, pHead2 #创建两个指针,分别指向两个链表的头节点#当指针指向的两个节点不相等:while p1 != p2:if p1:p1 = p1.next           #同时更新两个指针else:p1 = pHead2#直到p1走完了自己的节点,p2也走完了自己的节点,此时p1换到p2的这条道上;同理p2也走到p1的道上#两者还是每次都移动到下一个节点,最终p1会跟p2在某一个节点相遇,相遇的这个节点就是他们的公共节点!if p2:p2 = p2.nextelse:p2 = pHead1return p1                  #由于没有交点的时候,两个链表最后都会出现都是None值,所以此时返回的p1 = None          

这篇关于[牛客网刷题 Day2] JZ52 两个链表的第一个公共结点(没做出来)双指针巧解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

好题——hdu2522(小数问题:求1/n的第一个循环节)

好喜欢这题,第一次做小数问题,一开始真心没思路,然后参考了网上的一些资料。 知识点***********************************无限不循环小数即无理数,不能写作两整数之比*****************************(一开始没想到,小学没学好) 此题1/n肯定是一个有限循环小数,了解这些后就能做此题了。 按照除法的机制,用一个函数表示出来就可以了,代码如下

csu1329(双向链表)

题意:给n个盒子,编号为1到n,四个操作:1、将x盒子移到y的左边;2、将x盒子移到y的右边;3、交换x和y盒子的位置;4、将所有的盒子反过来放。 思路分析:用双向链表解决。每个操作的时间复杂度为O(1),用数组来模拟链表,下面的代码是参考刘老师的标程写的。 代码如下: #include<iostream>#include<algorithm>#include<stdio.h>#

poj1330(LCA最近公共祖先)

题意:求最近公共祖先 思路:之前学习了树链剖分,然后我就用树链剖分的一小部分知识就可以解这个题目了,记录每个结点的fa和depth。然后查找时,每次将depth大的结点往上走直到x = y。 代码如下: #include<iostream>#include<algorithm>#include<stdio.h>#include<math.h>#include<cstring>

深入手撕链表

链表 分类概念单链表增尾插头插插入 删尾删头删删除 查完整实现带头不带头 双向链表初始化增尾插头插插入 删查完整代码 数组 分类 #mermaid-svg-qKD178fTiiaYeKjl {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-

建立升序链表

题目1181:遍历链表 时间限制:1 秒 内存限制:32 兆 特殊判题:否 提交:2744 解决:1186 题目描述: 建立一个升序链表并遍历输出。 输入: 输入的每个案例中第一行包括1个整数:n(1<=n<=1000),接下来的一行包括n个整数。 输出: 可能有多组测试数据,对于每组数据, 将n个整数建立升序链表,之后遍历链表并输出。 样例输

两个月冲刺软考——访问位与修改位的题型(淘汰哪一页);内聚的类型;关于码制的知识点;地址映射的相关内容

1.访问位与修改位的题型(淘汰哪一页) 访问位:为1时表示在内存期间被访问过,为0时表示未被访问;修改位:为1时表示该页面自从被装入内存后被修改过,为0时表示未修改过。 置换页面时,最先置换访问位和修改位为00的,其次是01(没被访问但被修改过)的,之后是10(被访问了但没被修改过),最后是11。 2.内聚的类型 功能内聚:完成一个单一功能,各个部分协同工作,缺一不可。 顺序内聚:

【每日一题】LeetCode 2181.合并零之间的节点(链表、模拟)

【每日一题】LeetCode 2181.合并零之间的节点(链表、模拟) 题目描述 给定一个链表,链表中的每个节点代表一个整数。链表中的整数由 0 分隔开,表示不同的区间。链表的开始和结束节点的值都为 0。任务是将每两个相邻的 0 之间的所有节点合并成一个节点,新节点的值为原区间内所有节点值的和。合并后,需要移除所有的 0,并返回修改后的链表头节点。 思路分析 初始化:创建一个虚拟头节点

【C++学习笔记 20】C++中的智能指针

智能指针的功能 在上一篇笔记提到了在栈和堆上创建变量的区别,使用new关键字创建变量时,需要搭配delete关键字销毁变量。而智能指针的作用就是调用new分配内存时,不必自己去调用delete,甚至不用调用new。 智能指针实际上就是对原始指针的包装。 unique_ptr 最简单的智能指针,是一种作用域指针,意思是当指针超出该作用域时,会自动调用delete。它名为unique的原因是这个

C语言指针入门 《C语言非常道》

C语言指针入门 《C语言非常道》 作为一个程序员,我接触 C 语言有十年了。有的朋友让我推荐 C 语言的参考书,我不敢乱推荐,尤其是国内作者写的书,往往七拼八凑,漏洞百出。 但是,李忠老师的《C语言非常道》值得一读。对了,李老师有个官网,网址是: 李忠老师官网 最棒的是,有配套的教学视频,可以试看。 试看点这里 接下来言归正传,讲解指针。以下内容很多都参考了李忠老师的《C语言非

每日一题|牛客竞赛|四舍五入|字符串+贪心+模拟

每日一题|四舍五入 四舍五入 心有猛虎,细嗅蔷薇。你好朋友,这里是锅巴的C\C++学习笔记,常言道,不积跬步无以至千里,希望有朝一日我们积累的滴水可以击穿顽石。 四舍五入 题目: 牛牛发明了一种新的四舍五入应用于整数,对个位四舍五入,规则如下 12345->12350 12399->12400 输入描述: 输入一个整数n(0<=n<=109 ) 输出描述: 输出一个整数