链表经典算法OJ题目

1.单链表相关经典算OJ题目1:移除链表元素

思路一

直接在原链表里删除val元素,然后让val前一个结点和后一个节点连接起来。

这时我们就需要3个指针来遍历链表:

pcur  —— 判断节点的val值是否于给定删除的val值相等

prev ——保存pcur的前一个节点,为删除节点后,连接pcur之后的节点做准备

del —— 保存pcur之后的一个节点,为删除节点之后,连接链表做准备,和继续遍历链表

那么一开始它们的位置应该怎么放呢?

链表肯定是从第一个节点开始遍历的,那么 pcur 肯定就指向第一个节点,del可以先不给值,等到要删除的时候在把pcur之后的节点赋值给它

prev是在pcur前面的一个节点,我们会发现,当第一次遍历时,pcur前面并没有节点,那么我们就创造一个节点:哨兵卫(Sentinel),于链表连接,再让prev指向它。之后我们在返回新的链表的时候,我们返回 哨兵卫(Sentinel)的下一个节点即可。

代码 :

typedef struct ListNode ListNode;
struct ListNode* removeElements(struct ListNode* head, int val) {//创建哨兵卫节点并于链表连接ListNode* newhead = (ListNode*)malloc(sizeof(ListNode));newhead->next = head;ListNode* prev = newhead;ListNode* pcur = head;ListNode* del;当我们pcur走到尾节点指向的NULL处时停止循环while(pcur){//当符合给定的val值,删除if(pcur->val == val){del = pcur->next;   //记录pcur下一个节点位置free(pcur);         //删除pcur节点prev->next = del;   //将pcur前后的节点连接起来pcur = del;         //走向下一个节点}//当不符合给定的val值,继续遍历链表else{      prev = pcur;pcur = pcur->next;}}//释放不要的哨兵卫节点并返回新的链表ListNode* p = newhead->next;free(newhead);return p;
}

 思路二

直接创建一个新的链表,遍历原链表,将不是val值的节点尾插到新的链表中

需要用到两个指针:

pcur —— 用来遍历原链表,找不是val值的节点

ptail —— 新链表的尾节点,将来尾插至此节点后面

代码:

typedef struct ListNode ListNode;
struct ListNode* removeElements(struct ListNode* head, int val) {//创建哨兵卫节点,这里是为了防止对NULL指针解引用。ListNode* newhead = (ListNode*)malloc(sizeof(ListNode));newhead->next =  NULL;ListNode* newtail,* pcur;newtail = newhead;pcur = head;//当pcur走到原尾节点->NULL时停止循环while (pcur){//如果和给定的val值节点不相等,则尾插至新链表中if (pcur->val != val){newtail->next = pcur;newtail = newtail->next;}pcur = pcur->next;    //pcur继续遍历原链表}newtail->next = NULL;  //出了循环之后,记得把新的链表的尾节点指向NULL,保证不会带出多余节点//释放哨兵卫节点ListNode* p = newhead->next;free(newhead);return p;
}

 2.单链表相关经典算OJ题目2:反转链表

思路一

创建一个新的链表,将原链表的节点依次拿到新链表进行头插

需要两个指针:

pcur —— 遍历原链表

newhead —— 作为新链表的第一个节点

代码:

typedef struct ListNode ListNode;
struct ListNode* reverseList(struct ListNode* head) {//如果链表为NULL则不用反转if(head == NULL){return head;}ListNode* newhead = NULL;ListNode* pcur = head;ListNode* Transfer = NULL;//开始反转while(pcur){Transfer = pcur;   // 保存当前节点的指针pcur = pcur->next; // 移动到下一个节点Transfer->next = newhead;  // 反转当前节点的next指针,指向新的头节点newhead = Transfer;   // 更新新链表的头节点}return newhead;
}

思路二

创建三个指针,完成链表的反转:

n1 : 记录n2前一个节点

n2 : 用来遍历链表

n3 : 记录n2后一个节点

代码:

typedef struct ListNode ListNode;
struct ListNode* reverseList(struct ListNode* head) {//链表为NULL时不用反转if (head == NULL) {return head;}ListNode* n1 = NULL;ListNode* n2 = head;ListNode* n3 = n2->next;当n2遍历完链表时循环停止while (n2){n2->next = n1;   //让n2节点指向n1节点n1 = n2;         //让n1走至下一个节点n2 = n3;         //让n2走至下一个节点// 因为n3比n2快一步,当n2走至尾节点的时候,n3已经为NULL了,不处理的话会造成NULL的解引用if (n3 != NULL)   n3 = n3->next;   //让n3走至下一个节点}//当循环结束时n1会成为新的第一个节点return n1;}

 3.单链表相关经典算OJ题目3:链表的中间节点

思路 

这一题说难也不难,相信想一下大部分的码农都可以写出来

这里我就介绍一个更妙的方法:快慢指针法

什么是快慢指针法?顾名思义,就是创造两个指针,一个指针走的慢一点,一个指针走的快一点,对应到我们这一题上就是,同一时间内,一个指针走一步,一个指针走两步:2 × slow = fast

那么它是否可以应对题目给出的条件呢?当中间节点有两个时,返回第二个中间节点

两个中间节点

单中间节点

 

可以看见无论是双中间节点 还是 单中间节点这个方法都可以完美解决,我们也可以观察出循环的结束条件为:当 fast 指针为NULL 或 fast指针下一个节点为NULL时停止

代码:

typedef struct ListNode ListNode;
struct ListNode* middleNode(struct ListNode* head) {//创建快慢指针ListNode *slow,*fast = NULL;链表不为NULL时快慢指针从第一个节点开始遍历if(head != null);slow = fast = head;//当 fast 指针为NULL 或 fast指针下一个节点为NULL时停止//这两个表达式位置不可以换,若换,当fast为NULL时会出现对NULL指针解引用//如果换,当fast为NULL时,第一个表达式为假,第二个表达式就不执行了,不会出现NULL指针解引用while(fast != NULL && fast->next){slow = slow->next;        //慢指针走一个节点fast = fast->next->next;  //快指针走两个节点}return sl
}

  4.单链表相关经典算OJ题目4:合并两个有序数组

  

思路 

创建新链表,并且通过遍历两个原来的链表比较大小来添加至新的链表中

代码:

typedef struct ListNode ListNode;
struct ListNode* mergeTwoLists(struct ListNode* list1, struct ListNode* list2) {//设置哨兵节点当新链表的第一个节点ListNode* newhead = (ListNode*)malloc(sizeof(ListNode));newhead->next = NULL;ListNode* newtail = newhead; //只有一个节点,第一个节点 = 尾节点//创建l1 l2 指针遍历原链表list1 list2ListNode* l1 = list1;ListNode* l2 = list2;//当遍历完一个链表时,循环停止,不可能出现两个链表同时遍历完的情况while(l1 != NULL && l2 != NULL){//l1 或 l2 指向的值,小的放入新链表,并让其指针l1或l2 与 newtail指针往后走if(l1->val < l2->val){newtail->next = l1;l1 = l1->next;newtail = newtail->next;}else{newtail->next = l2;l2 = l2->next;newtail = newtail->next;}}//将没有遍历完的链表,将其内的元素一次性插入新链表中if(l1 != NULL){newtail->next = l1;}else{newtail->next = l2;}//存储哨兵卫后面的节点,并释放哨兵卫节点ListNode* p = newhead->next;free(newhead);return p;    //返回新的头节点
}

  5.单链表相关经典算OJ题目5:环形链表的约瑟夫问题

思路

利用我们的环形链表,依次的去遍历链表,使用count计数器计数,当count = m时删除节点,当环形链表只剩一个节点(这个节点下一个节点指向自己)时,就说明找到了最后留下的人。

这时我们需要用到两个指针:

pcur —— 遍历环形链表

prev —— 保存pcur的前一个节点,为删除节点做准备

代码:

typedef struct ListNode ListNode;//创建新节点
ListNode* Buynode(int x){ListNode* p = (ListNode*)malloc(sizeof(ListNode));p->val = x;p->next = NULL;return p;
}//创建环形链表
ListNode* creatCircle(int n){ListNode* head,*ptail;    head = Buynode(1);      //创建第一个节点val值为1ptail = head;for(int i = 2;i <= n;i++)     //继续创建节点,直到节点数 = n ,从第一个节点val = 1{                             //第二个 = 2....一直到n个节点,其val = nptail->next = Buynode(i);ptail = ptail->next;}// 让链表的头尾相连ptail->next = head;//因为要用到第一个节点的上一个节点,所以返回ptailreturn ptail;
}int ysf(int n, int m ) {ListNode* prev,*pcur;    //创建所需要的指针prev = creatCircle(n);   //创建环形链表pcur = prev->next;       //pcur从没有相连的第一个节点开始遍历int count = 1;           //创建计数器//开始循环,当环形链表中的节点指向自己时,说明只剩一个节点,循环结束while(pcur->next != pcur){//如果计数器 == m 则删除该节点if(count == m){prev->next = pcur->next;  //prev->next指向需删除节点的下一个节点free(pcur);               //删除该节点pcur = prev->next;        //pcur走向已删除节点的下一个节点count = 1;                //重置计数器}//如果计数器 != m 则删除该节点else{prev = pcur;            //两个指针都走向各自的下一个节点pcur = pcur->next;count++;                //计数器++}}return pcur->val;     //返回最后一个节点的val值
}

    6.单链表相关经典算OJ题目 6 :分割链表

思路一 

修改原链表,把小于特定的 x 值尾插到链表后面

pcur —— 遍历链表直到原尾节点

prev —— pcur的前一个节点,为了删除节点准备

patil —— 原尾节点

newpatil —— 新的尾节点

newhead —— 一个哨兵节点作为新的头节点

代码:

typedef struct ListNode ListNode;
struct ListNode* partition(struct ListNode* head, int x){//如果链表尾为NULL则不用分割if(head == NULL){return head;}//创建哨兵卫并于链表第一个节点连接,让它成为新的第一个节点ListNode* newhead = (ListNode*)malloc(sizeof(ListNode));newhead->next = head;//分割链表需要用到的指针ListNode* prev,*pcur,*ptail,*newptail;prev = newhead;pcur = head;//找尾节点while(pcur->next != NULL){pcur = pcur->next;}ptail = pcur;newptail = pcur;pcur = head;   //pcur重新回到原第一个节点//当pcur遍历到原尾节点的时候停止循环while(pcur != ptail){//如果节点的val值比x大,则把节点尾插至链表if(pcur->val >= x)   {prev->next = pcur->next;      //先让prev的下一个节点指向pcur下一个节点newptail->next = pcur;       //pcur节点尾插至链表后newptail = newptail->next;  //新的尾节点pcur = prev->next;         //让pcur走到prev指向的下一个节点}//否者pcur与prev各自往后走一个节点else{prev = pcur;pcur = pcur->next;}}newptail->next = NULL;断绝尾节点后面还带着其它节点ListNode* p = newhead->next;free(newhead);return p;
}

思路二

创建一个新的链表,把大于等于x的节点尾插到新的链表中,小于x的节点头插至链表中

 代码:

typedef struct ListNode ListNode;
struct ListNode* partition(struct ListNode* head, int x){if(head == NULL){return head;}//新链表起码要有一个节点,不然不知道是头插还是尾插ListNode* newhead,*pcur,*newptail,*del;newhead = newptail = pcur = head;   //把原链表的第一个节点给到新链表//这时新链表 头 = 尾 pcur = del = pcur->next;   //pcur走到原链表的第二个节点//当pcur走到NULL时说明已经遍历完原链表while(pcur){//尾插if(pcur->val >= x){if(del)                     //怕出现NULL指针解引用del = pcur->next;          //存储pcur下一个节点,不然待会找不到newptail->next = pcur;     //让新链表尾节点指向pcurnewptail = newptail->next;   //更习新链表的尾节点pcur = del;                 //pcur走向下一个节点}//头插else{if(del)                    //怕出现NULL指针解引用del = pcur->next;          //存储pcur下一个节点,不然待会找不到pcur->next = newhead;      //pcur的这个节点头插至新链表第一个节点处newhead = pcur;            //更新新链表的第一个节点pcur = del;                 //pcur走向下一个节点}}//让新链表的尾节点指向NULl,防止有别的节点被带上newptail->next = NULL;return newhead;
}

思路三:

创建 一大  一小 两个链表 ,把大于等于点放到大链表中,小于x的节点放到小链表中,当原链表的节点被全部放置完毕,把大链表链接到小链表后面。

 代码:

typedef struct ListNode ListNode;
struct ListNode* partition(struct ListNode* head, int x){//链表为NULL直接返回链表if(head == NULL){return head;}//创建大链表的第一个节点,和尾节点ListNode* Bnewhead,*Bnewptail;Bnewhead = Bnewptail = (ListNode*)malloc(sizeof(ListNode));//创建小链表的第一个节点,和尾节点ListNode* Snewhead,*Snewptail;Snewhead = Snewptail = (ListNode*)malloc(sizeof(ListNode));ListNode* pcur = head;   //遍历原链表指针while(pcur){//尾插至大链表if(pcur->val >= x){Bnewptail->next = pcur;       //将大于等于x的pcur节点尾插至大链表Bnewptail = Bnewptail->next;  //更新大链表尾节点}//尾插至小链表else{Snewptail->next = pcur;       //将小于x的pcur节点尾插至小链表Snewptail = Snewptail->next;   //更新小链表尾节点} pcur = pcur->next;              //让pcur继续遍历原链表下一个节点}Bnewptail->next = NULL;     //让大链表的尾节点的下一个节点指向NULL防止它带出其他节点Snewptail->next = Bnewhead->next;   //让小链表的尾节点的下一个节点指向大链表的第一个节点//释放动态申请的空间ListNode* ret= Snewhead->next;   //存储大小合并后的链表的第一个有效节点free(Bnewhead);                 free(Snewhead);Bnewhead = Snewhead = NULL;//返回新链表return ret;}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/993.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

C语言进阶|通讯录的实现

基于动态顺序表实现通讯录 C语言基础要求&#xff1a;结构体、动态内存管理、顺序表、文件操作 1、功能要求 至少能够存储100个人的通讯信息能够保存用户信息&#xff1a;名字、性别、年龄、电话、地址等增加联系人信息删除指定联系人查找制定联系人修改指定联系人显示联系人…

Spring Boot后端与Vue前端融合:构建高效旅游管理系统

作者介绍&#xff1a;✌️大厂全栈码农|毕设实战开发&#xff0c;专注于大学生项目实战开发、讲解和毕业答疑辅导。 &#x1f345;获取源码联系方式请查看文末&#x1f345; 推荐订阅精彩专栏 &#x1f447;&#x1f3fb; 避免错过下次更新 Springboot项目精选实战案例 更多项目…

设计模式- 建造者模式(Builder Pattern)结构|原理|优缺点|场景|示例

目录 设计模式&#xff08;分类&#xff09; 设计模式&#xff08;六大原则&#xff09; 创建型 工厂方法 抽象工厂模式 单例模式 建造者模式 原型模式 结构型 适配器模式 建造者模式&#xff08;Builder Pattern&#xff09;是一种创…

【网络编程】TCP流套接字编程(TCP实现回显服务器)

一.TCP流套字节相关API. Socket(既能给客户端使用,也能给服务器使用) 构造方法 基本方法: ServerSocket(只能给服务器使用) 构造方法: 基本方法: 二.TCP实现回显服务器. 客户端代码示例: package Demo2;import java.io.IOException; import java.io.InputStream; import j…

【AI 测试】自然语言处理(NLP)类项目如何测试

目录 NLP类项目需要的技能针对NLP领域大模型的评测任务及评价指标设计如何开展测试一、准确性评测实例二、稳定性评测实例三、效率评测实例四、执行测试注意事项以下是摘自某招聘网站的AI 测试工作职责: 负责NLP等领域大模型评测任务及评价指标的设计与制定;跟进业内大模型技…

一个char类型数字转字符串后引起的惨案

问题现象 嵌入式开发平台&#xff0c;交叉编译链比较老&#xff0c;还不支持 C11 的 to_string 写法&#xff0c;此时想通过模板方式&#xff0c;构造一个通用的 toString 接口&#xff0c;采用了 ostringstream &#xff0c;就出现了问题。 模板接口如下 template <type…

2 逻辑斯蒂回归(分类)

目录 1 理论 逻辑回归假设数据服从伯努利分布&#xff08;二分类&#xff09;,通过极大化似然函数的方法&#xff0c;运用梯度下降来求解参数&#xff0c;来达到将数据二分类的目的。 逻辑斯蒂回归&#xff08;Logistic Regression&#xff09;是一种用于解决分类问题的…

esp32-通过wifi使用timelib库同步时间(三)

库的安装 本文基于platformIO&#xff0c;安装较为简单如下图 实例代码 完整代码如下&#xff0c;如果时间获取超时请使用time1.aliyun.com获取时间。 /** Time_NTP.pde* Example showing time sync to NTP time source** This sketch uses the Ethernet library*/#include …

Kafka入门介绍+集群部署+简单使用

Kafka入门介绍集群部署简单使用 简介核心概念Broker&#xff08;服务节点/实例&#xff09;Producer&#xff08;生产者&#xff09;Topic&#xff08;主题&#xff09;Partition&#xff08;分区&#xff09;Consumer&#xff08;消费者&#xff09;和Consumer Group&#xff…

shell 脚本常用 逻辑判断符 与 || 或 ! 非

shell 脚本常用 逻辑判断符 文章目录 1、&& 逻辑与2、|| 逻辑或3、! 逻辑非 – 1、&& 逻辑与 只有两个都是真&#xff0c;结果才是真 如果第一个式子为false&#xff0c;则不会执行后面 2、|| 逻辑或 一真则真 如果左端为真&#xff0c;则右端不需要再进行…

超级好用的C++实用库之字节流解析器

概述 字节流解析器是一种软件组件&#xff0c;它负责将接收到的原始二进制数据&#xff08;字节流&#xff09;转换为有意义的信息结构或格式。在计算机网络、文件处理和数据通信中&#xff0c;字节流是最基本的数据传输形式&#xff0c;但这些原始字节对于应用程序通常是没有直…

对组合模式的理解

目录 一、场景1、题目描述 【[案例来源](https://kamacoder.com/problempage.php?pid1090)】2、输入描述3、输出描述4、输入示例5、输出示例 二、实现&#xff08;假的组合模式&#xff09;1、代码2、为什么上面的写法是假的组合模式&#xff1f; 三、实现&#xff08;真的组合…

文本生成任务的评价方法BLEU 和 ROUGE

BLEU 是 2002 年提出的&#xff0c;而 ROUGE 是 2003 年提出的。这两种指标虽然存在着一些问题&#xff0c;但是仍然是比较主流的评价指标。 BLUE BLEU 的全称是 Bilingual evaluation understudy&#xff0c;BLEU 的分数取值范围是 0&#xff5e;1&#xff0c;分数越接近1&a…

使用new 关键字调用函数,创建对象的过程中做了什么

使用new 关键字调用函数&#xff0c;创建对象的过程中做了什么 使用 new关键字创建对象的过程大致可以分为以下几个步骤&#xff1a; 创建空对象&#xff1a;首先&#xff0c;new操作符会创建一个空对象&#xff0c;这个对象的隐式原型__proto__属性会被设置为构造函数的显示原…

YOLOv9改进策略 | 细节创新篇 | 迭代注意力特征融合AFF机制创新RepNCSPELAN4

一、本文介绍 本文给大家带来的改进机制是AFF&#xff08;迭代注意力特征融合&#xff09;&#xff0c;其主要思想是通过改善特征融合过程来提高检测精度。传统的特征融合方法如加法或串联简单&#xff0c;未考虑到特定对象的融合适用性。iAFF通过引入多尺度通道注意力模块(我…

搭建vue3组件库(一):Monorepo项目搭建

Monorepo Monorepo 是一种项目代码管理方式&#xff0c;指单个仓库中管理多个项目&#xff0c;有助于简化代码共享、版本控制、构建和部署等方面的复杂性&#xff0c;并提供更好的可重用性和协作性。 pnpm pnpm 全称 performant npm&#xff0c;意思为 高性能的 npm。pnpm 由…

mysql面试题六(视图,存储过程,触发器)

目录 1.什么是视图 视图的特点 视图的创建与使用 视图的用途与优势 注意事项 2.什么是存储过程 存储过程的特点 存储过程的创建与调用 存储过程的使用场景 注意事项 3.什么是触发器 触发器的特点 触发器的创建与管理 触发器的使用场景 注意事项 1.什么是视图 在…

算法打卡day52|单调栈篇03| 84.柱状图中最大的矩形

算法题 Leetcode 84.柱状图中最大的矩形 题目链接:84.柱状图中最大的矩形 大佬视频讲解&#xff1a;84.柱状图中最大的矩形视频讲解 个人思路 这题和接雨水是相似的题目&#xff0c;原理上基本相同&#xff0c;也是可以用双指针和单调栈解决&#xff0c;只是有些细节不同。…

深度学习之目标检测从入门到精通——json转yolo格式

记录点&#xff1a; import json import osname2id {person:0,helmet:1,Fire extinguisher:2,Hook:3,Gas cylinder:4}def convert(img_size, box):dw 1./(img_size[0])dh 1./(img_size[1])x (box[0] box[2])/2.0 - 1y (box[1] box[3])/2.0 - 1w box[2] - box[0]h box…

23种设计模式之行为模式篇

三、行为模式 这类模式主要关注对象之间的通信&#xff0c;尤其是它们之间进行通信的方式和时机。 包括&#xff1a; 策略模式&#xff08;Strategy&#xff09;模板方法模式&#xff08;Template Method&#xff09;观察者模式&#xff08;Observer&#xff09;迭代器模式&…