环形链表相关的练习
创始人
2024-06-01 00:45:44
0

目录

一、相交链表

二、环形链表

三、环形链表 ||



一、相交链表

给你两个单链表的头节点 headAheadB ,请你找出并返回两个单链表相交的起始节点。如果两个链表不存在相交节点,返回 null

图示两个链表在节点 c1 开始相交:

 题目数据 保证 整个链式结构中不存在环。

注意,函数返回结果后,链表必须 保持其原始结构

代码实现

struct ListNode *getIntersectionNode(struct ListNode *headA, struct ListNode *headB) 
{// 1. 分别找到两个单链表的尾结点,并计算它们的长度struct ListNode *tailA = headA, *tailB = headB;int lenA = 1, lenB = 1;while (tailA->next != NULL){++lenA;tailA = tailA->next;}while (tailB->next != NULL){++lenB;tailB = tailB->next;}if (tailA != tailB)  // 如果两个链表不相交,则尾结点的地址不同{return NULL;}// 2. 让指向长链表的指针先走差距步int gap = abs(lenA - lenB);struct ListNode *longCur = headA, *shortCur = headB;if (lenA < lenB){longCur = headB;shortCur = headA;}for (int i = 0; i < gap; ++i){longCur = longCur->next;}// 3. 让 longCur 和 shortCur 同时向后走,直到找到相同地址的结点while (longCur != shortCur){longCur = longCur->next;shortCur = shortCur->next;}return longCur;
}


二、环形链表

给你一个链表的头节点 head ,判断链表中是否有环。

如果链表中有某个节点,可以通过连续跟踪 next 指针再次到达,则链表中存在环。 为了表示给定链表中的环,评测系统内部使用整数 pos 来表示链表尾连接到链表中的位置(索引从 0 开始)。注意:pos 不作为参数进行传递 。仅仅是为了标识链表的实际情况。

如果链表中存在环 ,则返回 true 。 否则,返回 false

示例 1

 输入:head = [3,2,0,-4], pos = 1

输出:true

解释:链表中有一个环,其尾部连接到第二个节点。

示例 2

 输入:head = [1,2], pos = 0

输出:true

解释:链表中有一个环,其尾部连接到第一个节点。

示例 3

 输入:head = [1], pos = -1

输出:false

解释:链表中没有环。

提示

  • 链表中节点的数目范围是 [0, 10^4]

  • -105 <= Node.val <= 105

  • pos-1 或者链表中的一个 有效索引

进阶:你能用 O(1)(即,常量)内存解决此问题吗?

代码实现一

bool hasCycle(struct ListNode *head) 
{struct ListNode* addr[10000] = { 0 };  // addr 是保存每个结点地址的指针数组int pos = 0;  // pos 始终是第一个未存放结点地址的数组下标struct ListNode* cur = head;while (cur != NULL){addr[pos++] = cur;// 检查 cur->next 是否指向之前的结点或自己for (int i = 0; i < pos; ++i) {if (addr[i] == cur->next){return true;}}cur = cur->next;}return false;  
}

代码实现二(快慢双指针)

bool hasCycle(struct ListNode *head)
{struct ListNode* slow = head;struct ListNode* fast = head;while (fast && fast->next)  // 如果链表不带环,则快指针先走到空或尾{slow = slow->next;fast = fast->next->next;if (slow == fast){return true;}}return false;
}

问题(前提是链表带环)

  1. 快慢指针从相同的起始位置出发,慢指针 slow 每次走一步,快指针 fast 每次走两步,请问这两个指针为什么一定会再次相遇?

    设当 slow 走到入环的第一个结点时,fast 距 slow y 步(0 <= y <= C,C 表示环的长度),然后 slow 走 x 步,fast 走 2x 步后,两个指针再次相遇,则有:2x - x = y,即 x = y

    y == 0,即当 slow 走到入环的第一个结点时,就和 fast 再次相遇了;y == C,即入环的第一个结点就是头结点,如示例 2

  2. 快慢指针从相同的起始位置出发,慢指针 slow 每次走一步,快指针 fast 每次走 n 步(n >= 3),请问这两个指针也一定会再次相遇吗?

    n * x - x = y,即 (n - 1)x = y ==> x = y / (n - 1)

    例如

     


三、环形链表 ||

给定一个链表的头节点 head ,返回链表开始入环的第一个节点。 如果链表无环,则返回 null

不允许修改 链表。

示例 1

 输入:head = [3,2,0,-4], pos = 1

输出:返回索引为 1 的链表节点

解释:链表中有一个环,其尾部连接到第二个节点。

示例 2

 输入:head = [1,2], pos = 0

输出:返回索引为 0 的链表节点

解释:链表中有一个环,其尾部连接到第一个节点。

示例 3

 输入:head = [1], pos = -1

输出:返回 null

解释:链表中没有环。

提示

  • 链表中节点的数目范围在范围 [0, 10^4]

  • -105 <= Node.val <= 105

  • pos 的值为 -1 或者链表中的一个有效索引

进阶:你是否可以使用 O(1) 空间解决此题?

代码实现一

struct ListNode *detectCycle(struct ListNode *head) 
{struct ListNode* addr[10000] = { 0 };int pos = 0;struct ListNode* cur = head;while (cur != NULL){addr[pos++] = cur;for (int i = 0; i < pos; ++i){if (addr[i] == cur->next){return addr[i];}}cur = cur->next;}  return NULL;
}

代码实现二

struct ListNode *detectCycle(struct ListNode *head)
{struct ListNode* slow = head;struct ListNode* fast = head;while (fast && fast->next){slow = slow->next;fast = fast->next->next;if (slow == fast)  // 再次相遇{struct ListNode* start = head;  // start 从起始结点出发struct ListNode* meet = slow;  // meet 从相遇结点出发while (start != meet){start = start->next;meet = meet->next;}return start;  // 或者 return meet;}}return NULL;
}

分析

其中 L 表示指针从头结点走到入环第一个结点所需要的步数,x 表示慢指针走到入环的第一个结点后,与快指针再次相遇所需走的步数,C 则表示环的长度

从起点出发到再次相遇,慢指针 slow 走的步数为 L + x,快指针 fast 走的步数为 L + k * C + x(k >= 1),又因为 slow 每次走一步,fast 每次走两步,所以有:2(L + x) = L + k * C + x,即 L = k * C + x ==> L = (k - 1)C + x

由此得出一个结论:在链表有环的前提下,一个指针从起始结点开始走,另一个指针从再相遇结点开始走,两个指针每次走一步,最终这两个结点会在入环的第一个结点相遇

代码实现三

struct ListNode *getIntersectionNode(struct ListNode *headA, struct ListNode *headB) 
{struct ListNode *tailA = headA, *tailB = headB;int lenA = 1, lenB = 1;while (tailA->next != NULL){++lenA;tailA = tailA->next;}while (tailB->next != NULL){++lenB;tailB = tailB->next;}if (tailA != tailB){return NULL;}int gap = abs(lenA - lenB);struct ListNode *longCur = headA, *shortCur = headB;if (lenA < lenB){longCur = headB;shortCur = headA;}for (int i = 0; i < gap; ++i){longCur = longCur->next;}while (longCur != shortCur){longCur = longCur->next;shortCur = shortCur->next;}return longCur;
}
​
struct ListNode *detectCycle(struct ListNode *head)
{struct ListNode* slow = head;struct ListNode* fast = head;while (fast && fast->next){slow = slow->next;fast = fast->next->next;if (slow == fast){// 转换成求相交结点struct ListNode* headB = slow->next;slow->next = NULL;return getIntersectionNode(head, headB);}}return NULL;
}

相关内容

热门资讯

智算中心“多地开花” 项目建设...   本报记者 寇佳丽  算力是数字经济时代的新质生产力。近日,珠海、济南、合肥等多地的智算中心项目正...
全球首个!可在复杂环境自主决策... 转自:千龙网当人形机器走进生产、生活,一个机器人能否实现“多能”?3月12日,北京人形机器人创新中心...
全国首发!北京海淀推出具身智能... 转自:千龙网3月12日,记者从北京市海淀区市场监管局获悉,海淀区近日发布全国首个面向具身智能技术在厨...
A股横盘震荡 泛AI主线强势不... 3月12日,A股市场呈现横盘震荡格局,有色金属、白酒等权重板块下挫,拖累三大股指收跌。AI主线热度不...
兰州公安发布一周典型电诈案件预... 【本报讯】3月12日,兰州市公安局反诈中心发布一周典型电诈案件预警,提醒广大市民防范刷单返利、冒充抖...
过敏性鼻炎用药需谨慎 【本报讯】春回大地,万物复苏,但恼人的鼻塞、喷嚏、流涕却让许多人谈“春”色变。日前,兰州市第一人民医...
市场潜力持续释放 2月份我国汽...   本报记者 陈 潇  3月11日,中国汽车工业协会信息发布会在北京召开。中国汽车工业协会副秘书长陈...
上交所发布行动方案 引导行业践...   本报讯 (记者毛艺融)3月12日,记者从上海证券交易所获悉,为深入贯彻党的二十届三中全会、中央金...
让青春之花绽放在祖国最需要的地... 在兰州市第十七中学的校园里,33岁的语文教师刘雨佳正用热情和坚持,书写着属于自己的青春篇章。 自20...
女人头发少适合什么发型 女人头发少适合什么发型头发少适合韩式小短发这是一款非常小清新的韩式小短发,浅亚麻色染发+短直发+空气...
环保农资进乡村 不误农时不误春 春耕时节,农资供应关系到全年农业生产的方方面面。3月11日,记者走进皋兰县各大农副日用品公司,发现皋...
童话般的故事 童话般的故事格林啊,一千零一夜啊
剑道妖尊的介绍 剑道妖尊的介绍《剑道妖尊》是木扬写的网络小说连载于奇迹网。
“工业味精”全球供应趋紧 预计... 由于价格迅速上涨、海内外价差较大,有“工业味精”之称的锑最近站在了市场的聚光灯下。上海钢联数据显示,...
跟金蟾有关的成语 跟金蟾有关的成语金蟾献瑞关于这个成语的故事和寓意:金蟾是民间传说中能吞吐金钱的灵物。民间流传着刘海戏...
作文听完笑话后 作文听完笑话后作文听完笑话后... 作文听完笑话后 展开今天,我闹了一个天大的笑话。早上,我吃完...
游戏人生怎么开通啊 游戏人生怎么开通啊荣誉大于500的好友可以邀你开,系统每天中午发放10000个邀请名额,今天的名额1...
中信消费金融回应增资获批:将进...   本报记者 李 冰  日前,国家金融监督管理总局北京监管局正式批复中信消费金融有限公司(以下简称“...
游戏人生怎么开啊? 游戏人生怎么开啊?你找一个有游戏人生的人,每天十点每个人可以邀请一个好友,要十点因为一万个名额一分钟...
想学小吃去哪里学? 想学小吃去哪里学?小吃行业现在属于火爆的热门行业,未来发展还是不错的。学小吃可以来专门教授,口碑好的...