数据结构与算法⑤(第二章OJ题,上)前五道链表面试题(下)

简介: 数据结构与算法⑤(第二章OJ题,上)前五道链表面试题

数据结构与算法⑤(第二章OJ题,上)前五道链表面试题(上):https://developer.aliyun.com/article/1513343

普通思路的代码:

struct ListNode* middleNode(struct ListNode* head){
    struct ListNode*cur=head;
    int count=0;
    while(cur!=NULL)
    {
        cur=cur->next;
        count++;//计算链表长度
    }
    struct ListNode*middle=head;
    for(int i=0;i<count/2;i++)//如5的话3次,6的话4次
    {
        middle=middle->next;
    }
    return middle;
}

这样的时间复杂度是N+N/2 虽然还是O(N)

但是以后面试可能会要求只能遍历一遍数组呢,这时就要用到快慢指针的思想了

虽然以前有用过,但是变一下就想不到了,看到这个思路又是震惊的一天...

运用快慢指针的代码:

struct ListNode* middleNode(struct ListNode* head) {
    struct ListNode* fast = head , * slow = head;
    while (fast!=NULL&&fast->next!=NULL)
    {
        fast=fast->next->next;//快指针一次走两步
        slow=slow->next;//慢指针一次走一步
    }
    return slow;
}

4. 输出该链表中倒数第k个结点

剑指 Offer 22. 链表中倒数第k个节点

输入一个链表,输出该链表中倒数第k个节点。为了符合大多数人的习惯,

本题从1开始计数,即链表的尾节点是倒数第1个节点。

例如,一个链表有 6 个节点,从头节点开始,它们的值依次是 1、2、3、4、5、6。

这个链表的倒数第 3 个节点是值为 4 的节点。

示例:

给定一个链表: 1->2->3->4->5, 和 k = 2.

返回链表 4->5.

/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     struct ListNode *next;
 * };
 */
struct ListNode* getKthFromEnd(struct ListNode* head, int k){
 
}

(这题没给范围,有点不严谨,但是懂思路就好了)

力扣这题是后更新的,所以测试用例更不严谨

这里说一下,力扣应该是OJ题做的最好的,但是面试那些应该用牛客多一点。

牛客网链接:https://www.nowcoder.com/practice/529d3ae5a407492994ad2a246518148a?tpId=13&&tqId=11167&rp=2&ru=/activity/oj&qru=/ta/coding-interviews/question-ranking

普通思路的代码:

这题和上一题的普通思路一样,相当于遍历两次链表

struct ListNode* getKthFromEnd(struct ListNode* head, int k){
    struct ListNode*cur=head;
    int count=0;
    while(cur!=NULL)
    {
        cur=cur->next;
        count++;//计算链表长度
    }
    struct ListNode*newHead=head;
    for(int i=0;i<count-k;i++)
    {
        newHead=newHead->next;
    }
    return newHead;
}

写完普通思路我想着用快慢指针怎么用,想了几秒就觉得不行就不想了,以为不能用

(不想动太多脑的坏处)看到别人的思路我已经开始咬嘴唇了...


虽然觉得k不一样时,时间一样,但是帅就对了:

运用快慢指针的代码:

struct ListNode* getKthFromEnd(struct ListNode* head, int k){
    struct ListNode* fast = head , * slow = head;
    while(k--)
    {
        if(fast==NULL)
        {
            return NULL;
        }
        fast=fast->next;//快指针先走k步
    }
    while(fast)
    {
        fast=fast->next;
        slow=slow->next;//快慢指针一起走
    }
    return slow;
}

5. 合并两个有序链表为一个新的有序链表

21. 合并两个有序链表

难度简单

将两个升序链表合并为一个新的 升序 链表并返回。

新链表是通过拼接给定的两个链表的所有节点组成的。

示例 1:



输入:l1 = [1,2,4], l2 = [1,3,4]

输出:[1,1,2,3,4,4]

示例 2:

输入:l1 = [], l2 = []

输出:[]

示例 3:

输入:l1 = [], l2 = [0]

输出:[0]

提示:

  • 两个链表的节点数目范围是 [0, 50]
  • -100 <= Node.val <= 100
  • l1 和 l2 均按 非递减顺序 排列
/**
 * Definition for singly-linked list.
 * struct ListNode {
 *     int val;
 *     struct ListNode *next;
 * };
 */
struct ListNode* mergeTwoLists(struct ListNode* list1, struct ListNode* list2){
 
}

尾插法代码:

 struct ListNode* mergeTwoLists(struct ListNode* list1, struct ListNode* list2) {
     if (list1 == NULL)
     {
         return list2;
     }
     if (list2 == NULL)
     {
         return list1;
     }
     struct ListNode* cur=NULL;
     if(list1->val <= list2->val)
     {
        cur = list1;
        list1 = list1->next;
     }
     else
     {
        cur = list2;
        list2 = list2->next;
     }
     struct ListNode* new = cur;
     while (list1 && list2)
     {
         if (list1->val <= list2->val)
         {
             cur->next = list1;
             cur = list1;
             list1 = list1->next;
         }
         else
         {
             cur->next = list2;
             cur = list2;
             list2 = list2->next;
         }
     }
     if (list1 == NULL)
     {
         cur->next = list2;
     }
     else
     {
         cur->next = list1;
     }
     return new;
}

尾插法+哨兵位的头节点代码:

哨兵位的头节点就是不存数据的节点

 struct ListNode* mergeTwoLists(struct ListNode* list1, struct ListNode* list2) {
     if (list1 == NULL)
     {
         return list2;
     }
     if (list2 == NULL)
     {
         return list1;
     }
     struct ListNode* cur = (struct ListNode*)malloc(sizeof(struct ListNode));
     struct ListNode* new = cur;
     while (list1 && list2)
     {
         if (list1->val <= list2->val)
         {
             cur->next = list1;
             cur = list1;
             list1 = list1->next;
         }
         else
         {
             cur->next = list2;
             cur = list2;
             list2 = list2->next;
         }
     }
     if (list1 == NULL)
     {
         cur->next = list2;
     }
     else
     {
         cur->next = list1;
     }
     struct ListNode* newHead = new->next;
     free(new);
     return newHead;
}

本篇完。

后面还有接着这部分的较难的OJ题

目录
相关文章
|
2月前
|
存储 算法 Perl
数据结构实验之链表
本实验旨在掌握线性表中元素的前驱、后续概念及链表的建立、插入、删除等算法,并分析时间复杂度,理解链表特点。实验内容包括循环链表应用(约瑟夫回环问题)、删除单链表中重复节点及双向循环链表的设计与实现。通过编程实践,加深对链表数据结构的理解和应用能力。
63 4
|
8天前
|
数据库
数据结构中二叉树,哈希表,顺序表,链表的比较补充
二叉搜索树,哈希表,顺序表,链表的特点的比较
数据结构中二叉树,哈希表,顺序表,链表的比较补充
|
2月前
|
存储 缓存 算法
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式
在C语言中,数据结构是构建高效程序的基石。本文探讨了数组、链表、栈、队列、树和图等常见数据结构的特点、应用及实现方式,强调了合理选择数据结构的重要性,并通过案例分析展示了其在实际项目中的应用,旨在帮助读者提升编程能力。
68 5
|
2月前
|
存储 C语言
【数据结构】手把手教你单链表(c语言)(附源码)
本文介绍了单链表的基本概念、结构定义及其实现方法。单链表是一种内存地址不连续但逻辑顺序连续的数据结构,每个节点包含数据域和指针域。文章详细讲解了单链表的常见操作,如头插、尾插、头删、尾删、查找、指定位置插入和删除等,并提供了完整的C语言代码示例。通过学习单链表,可以更好地理解数据结构的底层逻辑,提高编程能力。
114 4
|
2月前
|
算法 安全 搜索推荐
2024重生之回溯数据结构与算法系列学习之单双链表精题详解(9)【无论是王道考研人还是IKUN都能包会的;不然别给我家鸽鸽丢脸好嘛?】
数据结构王道第2.3章之IKUN和I原达人之数据结构与算法系列学习x单双链表精题详解、数据结构、C++、排序算法、java、动态规划你个小黑子;这都学不会;能不能不要给我家鸽鸽丢脸啊~除了会黑我家鸽鸽还会干嘛?!!!
|
2月前
|
算法
数据结构之购物车系统(链表和栈)
本文介绍了基于链表和栈的购物车系统的设计与实现。该系统通过命令行界面提供商品管理、购物车查看、结算等功能,支持用户便捷地管理购物清单。核心代码定义了商品、购物车商品节点和购物车的数据结构,并实现了添加、删除商品、查看购物车内容及结算等操作。算法分析显示,系统在处理小规模购物车时表现良好,但在大规模购物车操作下可能存在性能瓶颈。
53 0
|
2月前
|
C语言
【数据结构】双向带头循环链表(c语言)(附源码)
本文介绍了双向带头循环链表的概念和实现。双向带头循环链表具有三个关键点:双向、带头和循环。与单链表相比,它的头插、尾插、头删、尾删等操作的时间复杂度均为O(1),提高了运行效率。文章详细讲解了链表的结构定义、方法声明和实现,包括创建新节点、初始化、打印、判断是否为空、插入和删除节点等操作。最后提供了完整的代码示例。
78 0
|
9天前
|
机器学习/深度学习 算法
基于改进遗传优化的BP神经网络金融序列预测算法matlab仿真
本项目基于改进遗传优化的BP神经网络进行金融序列预测,使用MATLAB2022A实现。通过对比BP神经网络、遗传优化BP神经网络及改进遗传优化BP神经网络,展示了三者的误差和预测曲线差异。核心程序结合遗传算法(GA)与BP神经网络,利用GA优化BP网络的初始权重和阈值,提高预测精度。GA通过选择、交叉、变异操作迭代优化,防止局部收敛,增强模型对金融市场复杂性和不确定性的适应能力。
139 80
|
3天前
|
机器学习/深度学习 算法
基于遗传优化的双BP神经网络金融序列预测算法matlab仿真
本项目基于遗传优化的双BP神经网络实现金融序列预测,使用MATLAB2022A进行仿真。算法通过两个初始学习率不同的BP神经网络(e1, e2)协同工作,结合遗传算法优化,提高预测精度。实验展示了三个算法的误差对比结果,验证了该方法的有效性。
|
5天前
|
机器学习/深度学习 数据采集 算法
基于PSO粒子群优化的CNN-GRU-SAM网络时间序列回归预测算法matlab仿真
本项目展示了基于PSO优化的CNN-GRU-SAM网络在时间序列预测中的应用。算法通过卷积层、GRU层、自注意力机制层提取特征,结合粒子群优化提升预测准确性。完整程序运行效果无水印,提供Matlab2022a版本代码,含详细中文注释和操作视频。适用于金融市场、气象预报等领域,有效处理非线性数据,提高预测稳定性和效率。