欢迎来到尧图网

客户服务 关于我们

您的位置:首页 > 科技 > 能源 > 【链表OJ】常见面试题 2

【链表OJ】常见面试题 2

2024/10/24 4:30:12 来源:https://blog.csdn.net/2303_79015671/article/details/140938145  浏览:    关键词:【链表OJ】常见面试题 2

文章目录

  • 1.[链表分割](https://www.nowcoder.com/practice/0e27e0b064de4eacac178676ef9c9d70?tpId=8&&tqId=11004&rp=2&ru=/activity/oj&qru=/ta/cracking-the-coding-interview/question-ranking)
    • 1.1 题目要求
    • 1.2 哨兵位法
  • 2.[链表的回文结构](https://www.nowcoder.com/practice/d281619e4b3e4a60a2cc66ea32855bfa?tpId=49&&tqId=29370&rp=1&ru=/activity/oj&qru=/ta/2016test/question-ranking)
    • 2.1 题目要求
    • 2.2 快慢指针加反转链表
  • 3.[相交链表](https://leetcode.cn/problems/intersection-of-two-linked-lists/description/)
    • 3.1 题目要求
    • 3.2 双指针消除长度差
    • 3.3 哈希法
  • 4.[环形链表](https://leetcode.cn/problems/linked-list-cycle/description/)
    • 4.1 题目要求
    • 4.2 快慢指针

1.链表分割

链表分割

1.1 题目要求

现有一链表的头指针 ListNode* pHead,给一定值x,编写一段代码将所有小于x的结点排在其余结点之前,且不能改变原来的数据顺序,返回重新排列后的链表的头指针。

1.2 哨兵位法

创建两个哨兵位节点,一个用来存放val小于x的节点,一个存放val大于等于x的节点。
因为我们是顺序遍历,不会打乱原来的数据顺序,满足条件直接按要求放就可以了。最后再把存放val大于等于x的链表接到val小于x的链表后面就可以了。
但是最后会有一个坑!
当我们把两个链表连接后,可不能忘了head2(存放val大于等于x的节点)的最后一个节点可能不是指向NULL,就可能构成一个环,导致程序出错。
为什么会造成这种情况呢?
因为我们把节点链接到相应链表时没有除了节点的next,虽然后面会通过tail来处理next链接的问题,但是最后一个节点是做不到的。解决方法就是在最后处理一下,把tail2的next置为NULL就解决问题了。

class Partition {
public:ListNode* partition(ListNode* pHead, int x) {// write code hereListNode* head1 = (ListNode*)malloc(sizeof(ListNode));ListNode* head2 = (ListNode*)malloc(sizeof(ListNode));ListNode* tail1 = head1;ListNode* tail2 = head2;ListNode* cur = pHead;while(cur){if(cur->val<x){tail1->next = cur;tail1 = tail1->next;}else{tail2->next = cur;tail2 = tail2->next;}cur = cur->next;}tail1->next = head2->next;tail2->next = NULL;return head1->next;}
};

2.链表的回文结构

链表地回文结构

2.1 题目要求

判断链表是否是回文链表,是返回true,不是返回false。

2.2 快慢指针加反转链表

因为这个链表是单向的,无法做到像字符串那样,从两边往中间遍历来确定是否回文。
那么既然要判断链表是否的回文链表,肯定要先找到中间啊,找到中间就能找到两条相同的链表,你需要管节点数是单数的情况,中间的节点是不会影响最后的结果的。
在找到中间节点时,要记得把让中间节点的前前一个节点的next指向NULL。方便后续的比较。
通过快慢指针我们找到了链表的中间,但是怎么比较的,单链表可不能向前遍历。有什么办法吗?
当然了,让链表反转不就好了,这样的话就方便比较了,我们把链表的后一半反转,然后拿到反转后的头节点。
最后就是遍历比较了,一旦出现不同就返回false,都相同则返回true。
快慢指针加反转链表

class PalindromeList {
public:bool chkPalindrome(ListNode* A) {// write code here//先利用快慢指针法找到链表的中间//然后利用链表的反转将后半部分反转//最后在开始比较ListNode* fast = A;ListNode* slow = A;ListNode* prev = NULL;while(fast&&fast->next){fast = fast->next->next;prev = slow;slow = slow->next;}//slow即为链表中间//开始反转prev->next = NULL;prev = NULL;ListNode* cur = slow;while(cur){ListNode* next = cur->next;cur->next = prev;prev = cur;cur = next;}ListNode* l1 = A;ListNode* l2 = prev;while(l1&&l2){if(l1->val!=l2->val)return false;l1 = l1->next;l2 = l2->next;}return true;}
};

3.相交链表

相交链表

3.1 题目要求

找到A,B的第一个共同节点并返回,没有就返回NULL

3.2 双指针消除长度差

这里我借用题解里的一位大佬画的图大佬题解
双指针
有了这张图片,相信也不用太多解释。

struct ListNode *getIntersectionNode(struct ListNode *headA, struct ListNode *headB) {struct ListNode* a = headA;struct ListNode* b = headB;while(a!=b){a = a!=NULL?a->next:headB;b = b!=NULL?b->next:headA;}return a;
}

3.3 哈希法

其实这太题还有一种解法,哈希法,但是用C语言就比较不好写了。感兴趣的话,可以看一下下面的c++代码。

class Solution {
public:ListNode *getIntersectionNode(ListNode *headA, ListNode *headB) {unordered_set<ListNode *> visited;ListNode *temp = headA;while (temp != nullptr) {visited.insert(temp);temp = temp->next;}temp = headB;while (temp != nullptr) {if (visited.find(temp)!=visited.end()) {return temp;}temp = temp->next;}return nullptr;}
};

4.环形链表

环形指针

4.1 题目要求

找出链表中存在的环,如果存在就返回true,不存在就返回false

4.2 快慢指针

利用快慢指针,如果不存在环的话,慢指针永远也追不上快指针,直到快指针走到链表的尽头。
但是如果存在环的话,当慢指针还没进入环时,快指针肯定已经在环里面不断地循环了,而环里面是没有前后之分的,一旦慢指针进入环内,现在我们先想象这两个指针不是跳跃似地运动,而是平移,这样的话,快指针一定是会与慢指针相遇的。
快慢指针

可是如果是跳跃似地这样呢?
也就是为什么快指针每次走两步,慢指针走一步可以?
假设链表带环,两个指针最后都会进入环,快指针先进环,慢指针后进环。当慢指针刚
进环时,可能就和快指针相遇了,最差情况下两个指针之间的距离刚好就是环的长度。
此时,两个指针每移动一次,之间的距离就缩小一步,不会出现每次刚好是套圈的情
况,因此:在满指针走到一圈之前,快指针肯定是可以追上慢指针的,即相遇。
大家也可让快指针走3步看看行不行

bool hasCycle(struct ListNode *head) {struct ListNode* fast = head;struct ListNode* slow = head;while(fast&&fast->next){fast = fast->next->next;slow = slow->next;if(slow == fast)return true;}return false;
}

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com