<数据结构>五道LeetCode链表题分析.环形链表,反转链表,合并链表,找中间节点.

简介: <数据结构>五道LeetCode链表题分析.环形链表,反转链表,合并链表,找中间节点

一.反转链表


1.头插法

//头插法
struct ListNode* reverseList(struct ListNode* head)
{
    if(head==NULL)
    {
        return NULL;
    }
    struct ListNode*newhead=NULL;
    struct ListNode*next=head->next;
    while(head!=NULL)
    {
         next=head->next;
         head->next=newhead;//将head头查到newhead上
         newhead=head;//newhead为新的头
         head=next;  //head 接着往下走
    }
    return newhead;
}

2.迭代法

//迭代法
struct ListNode* reverseList(struct ListNode* head)
{
 
    if(head==NULL||head->next==NULL)
    {
        return head;
    }
    struct ListNode*n1=NULL; //利用三个指针 循环往下走
    struct ListNode*n2=head; //来调转链表指向
    struct ListNode*n3=n2->next; 
    while(n3->next!=NULL)
    {
        n2->next=n1;
        n1=n2;
        n2=n3;
        n3=n2->next;
    }
    n2->next=n1;
    n3->next=n2;
    return n3;
}

二.链表的中间节点


1.快慢指针法

//快慢指针法:fast指针一次走两步,slow指针一次走一步,
//          当fast为NULL或fast->next为NULL时(分奇偶),此时的slow为mid
struct ListNode* middleNode(struct ListNode* head)
{
    struct ListNode*slow=head;
    struct ListNode*fast=head;
    while(fast!=NULL&&fast->next!=NULL)
    {
        fast=fast->next->next;
        slow=slow->next;
    }
    return slow;
}

2.指针数组法

//不推荐此方法 空间复杂度为O(N)
struct ListNode* middleNode(struct ListNode* head)
{
   struct ListNode*arr[100]={NULL};
    int count=0;
    struct ListNode*p1=head;
    while(p1!=NULL)
    {
        arr[count]=p1;
        count++;
        p1=p1->next;
    }
    return arr[count/2];
 
}

三.合并两个有序链表


尾插法

//尾插法:创建一个新的头,比较大小,小的尾插到后面
struct ListNode* mergeTwoLists(struct ListNode* list1, struct ListNode* list2)
{
    if(list1==NULL)             // 特殊情况
       return list2;
    if(list2==NULL)
       return list1;
    struct ListNode*head=NULL;
    struct ListNode*tail=NULL;
    if(list1->val<=list2->val) //先把第一个给head和tail 保证不为空
    {
 
        tail=list1;
        head=list1;
        list1=list1->next;
    }
    else
    {
        tail=list2;
        head=list2;
        list2=list2->next;
    }
    while(list1!=NULL&&list2!=NULL)  //比较大小 小的尾插
    {
        if(list1->val<=list2->val)
        {
            tail->next=list1;
            tail=tail->next;
            list1=list1->next;
        }
        else
        {
 
            tail->next=list2;
            tail=tail->next;
            list2=list2->next;
        }
    }
    if(list1==NULL)              //最后有一个到头,直接接上后面
    {
        tail->next=list2;
    }
    else
    {
        tail->next=list1;
    }
    return head;
}

四.环形链表(1)


快慢指针法

//先让fast进入循环,然后slow进入,最后fast与slow相遇
bool hasCycle(struct ListNode *head)
{
    struct ListNode *slow=head;
    struct ListNode *fast=head;
    while(fast!=NULL&&fast->next!=NULL)
    {
        slow=slow->next;
        fast=fast->next->next;
        if(fast==slow)
        {
            return true;
        }
    }
    return false;
}

五.环形链表(2)


思路分析:

代码实现:

//快慢指针
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(fast==slow)                 //先快慢指针相遇
        {
            struct ListNode *meet=slow;
            while(meet!=head)
            {
                meet=meet->next;       //再一个从meet走,一个从head走
                head=head->next;
            }
            return head;               //相遇时为环形链表的入口
        }
    }
    return NULL;
 
}
目录
相关文章
|
7月前
|
机器学习/深度学习 算法
24. 两两交换链表中的节点, 19.删除链表的倒数第N个节点 ,面试题 02.07. 链表相交
1. **两两交换链表中的节点**:通过引入虚拟头结点,使所有节点都能采用统一的交换逻辑,避免对头结点单独处理。 2. **删除链表的倒数第N个节点**:利用双指针技巧,让快慢指针保持N个节点的距离,当快指针到达末尾时,慢指针正好指向待删除节点的前一个节点。 3. **链表相交**:先计算两链表长度并调整起点,确保从相同距离末尾的位置开始遍历,从而高效找到相交节点或确定无交点。 以上方法均在时间复杂度和空间复杂度上进行了优化,适合用于理解和掌握链表的基本操作及常见算法设计思路。
|
并行计算 算法 测试技术
C语言因高效灵活被广泛应用于软件开发。本文探讨了优化C语言程序性能的策略,涵盖算法优化、代码结构优化、内存管理优化、编译器优化、数据结构优化、并行计算优化及性能测试与分析七个方面
C语言因高效灵活被广泛应用于软件开发。本文探讨了优化C语言程序性能的策略,涵盖算法优化、代码结构优化、内存管理优化、编译器优化、数据结构优化、并行计算优化及性能测试与分析七个方面,旨在通过综合策略提升程序性能,满足实际需求。
500 1
LeetCode第二十四题(两两交换链表中的节点)
这篇文章介绍了LeetCode第24题的解法,即如何通过使用三个指针(preNode, curNode, curNextNode)来两两交换链表中的节点,并提供了详细的代码实现。
164 0
LeetCode第二十四题(两两交换链表中的节点)
Leetcode第十九题(删除链表的倒数第N个节点)
LeetCode第19题要求删除链表的倒数第N个节点,可以通过快慢指针法在一次遍历中实现。
212 0
Leetcode第十九题(删除链表的倒数第N个节点)
(剑指offer)18、删除链表的节点—22、链表中倒数第K个节点—25、合并两个排序的链表—52、两个链表的第一个公共节点(2021.12.07)
(剑指offer)18、删除链表的节点—22、链表中倒数第K个节点—25、合并两个排序的链表—52、两个链表的第一个公共节点(2021.12.07)
188 0
【LeetCode 46】450.删除二叉搜索树的节点
【LeetCode 46】450.删除二叉搜索树的节点
176 0
|
Unix Shell Linux
LeetCode刷题 Shell编程四则 | 194. 转置文件 192. 统计词频 193. 有效电话号码 195. 第十行
本文提供了几个Linux shell脚本编程问题的解决方案,包括转置文件内容、统计词频、验证有效电话号码和提取文件的第十行,每个问题都给出了至少一种实现方法。
308 6
LeetCode刷题 Shell编程四则 | 194. 转置文件 192. 统计词频 193. 有效电话号码 195. 第十行
|
Python
【Leetcode刷题Python】剑指 Offer 32 - III. 从上到下打印二叉树 III
本文介绍了两种Python实现方法,用于按照之字形顺序打印二叉树的层次遍历结果,实现了在奇数层正序、偶数层反序打印节点的功能。
185 6
|
搜索推荐 索引 Python
【Leetcode刷题Python】牛客. 数组中未出现的最小正整数
本文介绍了牛客网题目"数组中未出现的最小正整数"的解法,提供了一种满足O(n)时间复杂度和O(1)空间复杂度要求的原地排序算法,并给出了Python实现代码。
414 2

热门文章

最新文章