【面试经典 150 | 链表】删除链表的倒数第 N 个结点

devtools/2024/9/24 12:33:11/

文章目录

  • 写在前面
  • Tag
  • 题目来源
  • 解题思路
    • 方法一:统计节点个数
    • 方法二:双指针
  • 写在最后

写在前面

本专栏专注于分析与讲解【面试经典150】算法,两到三天更新一篇文章,欢迎催更……

专栏内容以分析题目为主,并附带一些对于本题涉及到的数据结构等内容进行回顾与总结,文章结构大致如下,部分内容会有增删:

  • Tag:介绍本题牵涉到的知识点、数据结构;
  • 题目来源:贴上题目的链接,方便大家查找题目并完成练习;
  • 题目解读:复述题目(确保自己真的理解题目意思),并强调一些题目重点信息;
  • 解题思路:介绍一些解题思路,每种解题思路包括思路讲解、实现代码以及复杂度分析;
  • 知识回忆:针对今天介绍的题目中的重点内容、数据结构进行回顾总结。

Tag

链表-删除节点】【迭代


题目来源

19. 删除链表的倒数第 N 个结点


解题思路

方法一:统计节点个数

思路

一种朴素的方法是首先统计链表中节点的个数 m,倒数第 n 个节点就是正数第 m - n + 1 个节点。我们从头结点开始,找到正数第 m - n 个节点(要删除节点的前一个节点),直接将该节点的指针指向下下个节点即可。

代码

/*** Definition for singly-linked list.* struct ListNode {*     int val;*     ListNode *next;*     ListNode() : val(0), next(nullptr) {}*     ListNode(int x) : val(x), next(nullptr) {}*     ListNode(int x, ListNode *next) : val(x), next(next) {}* };*/
class Solution {
public:const int getLength(ListNode* head) {int len = 0;while (head) {++len;head = head->next;}return len;}ListNode* removeNthFromEnd(ListNode* head, int n) {ListNode* dummy = new ListNode(0, head);int m = getLength(head);ListNode* cur = dummy;for (int i= 0; i < m - n; ++i) {cur = cur->next;}cur->next = cur->next->next;return dummy->next;}
};

复杂度分析

时间复杂度: O ( m ) O(m) O(m) m m m 为链表的长度。

空间复杂度: O ( 1 ) O(1) O(1)

方法二:双指针

思路

我们先用一个指针指向链表的第 n 个节点,此时增加另一个指针指向头结点,接着让两个指针同时向链表尾部移动,每一移动一个位置,当第一个指针到达 nullptr 时,我们也就找到了需要删除的节点。

如果我们利用 代码 中的 while 循环就找到了要删除节点的上一个节点,此时直接将该节点的指针指向下下个节点即可。

代码

/*** Definition for singly-linked list.* struct ListNode {*     int val;*     ListNode *next;*     ListNode() : val(0), next(nullptr) {}*     ListNode(int x) : val(x), next(nullptr) {}*     ListNode(int x, ListNode *next) : val(x), next(next) {}* };*/
class Solution {
public:ListNode* removeNthFromEnd(ListNode* head, int n) {ListNode *first = head;ListNode *dummy = new ListNode(0, head);ListNode *second = dummy;for(int i = 0; i < n; ++i)first = first->next;while(first){first = first->next;second = second->next;}second->next = second->next->next;return dummy->next;}
};

复杂度分析

时间复杂度: O ( m ) O(m) O(m) m m m 为链表的长度。

空间复杂度: O ( 1 ) O(1) O(1)


写在最后

如果您发现文章有任何错误或者对文章有任何疑问,欢迎私信博主或者在评论区指出 💬💬💬。

如果大家有更优的时间、空间复杂度的方法,欢迎评论区交流。

最后,感谢您的阅读,如果有所收获的话可以给我点一个 👍 哦。


http://www.ppmy.cn/devtools/12075.html

相关文章

H5 台球猜位置小游戏

刷到抖音有人这样玩&#xff0c;就写了一个这样的小游戏练习一下H5的知识点。 小游戏预览 w(&#xff9f;Д&#xff9f;)w 不开挂越急越完成不了&#xff0c;&#x1f47f;确认15次也没全对… 知识点 获取坐标位置的DOM元素&#xff0c;感觉应该是新的吧&#xff0c;以前的…

第 2 章:FFmpeg简介

2.1 历史 历史 一些相关术语介绍&#xff1a; 容器&#xff08;Container&#xff09;格式&#xff1a;一种文件封装格式&#xff0c;里边主要包含了流&#xff0c;一般会使用一个特定的后缀名标识&#xff0c;例如.mov、.avi、.wav等。流 &#xff08;Stream&#xff09;&am…

Hive,Presto,Spark 共性

Hive、Presto 和 Spark 都是大数据处理工具&#xff0c;都属于大数据处理技术栈&#xff0c;都需要集群环境支持&#xff0c;都可以进行数据处理和分析。 都可以进行数据处理&#xff1a;Hive、Presto、Spark 都可以用 SQL 语句进行数据处理&#xff0c;也可以用它们的语言&…

uniapp:小白1分钟学会使用webSocket(可无脑复制)

uni.connectSocket() uni.$emit页面通信 项目中使用uni.connectSocket()创建webSocket的总结&#xff0c;代码可无脑复制&#xff0c;直接使用。 1、main.js 引入vuex import store from ./store; Vue.prototype.$store store;vuex中封装webSocket 2、vuex的&#xff1a;index…

springcloud alibaba 整合seata的TCC

一、seata服务端搭建同上篇。 Seata的AT模式客户端两阶段提交流程源码分析 二、seata客户端的结构 1.示例DEMO工程 下单&#xff0c;扣余额&#xff0c; 减库存。 2. MAVEN配置。 父工程&#xff1a;由于spring-cloud-starter-alibaba-seata依赖的seata-spring-boot-starter…

【数据库】MySQL分页查询

分页查询&#xff1a; 数据记录条数过多的时候&#xff0c;需要分页来显示。 语法&#xff1a; select 查询字段 from 表名 where ....等等 limit offset&#xff08;开始记录索引,是从0开始的&#xff09;,size&#xff08;要取出的条数&#xff09;&#xff1b; 案例&…

如何使用WEB前端模板

我最近想搞一搞前端&#xff0c;前端属实不太行&#xff0c;像前端搞个模板直接套一下。但是发现下载下来也有点不知道怎么用起来&#xff0c;这里我就把我的一个Bootstrap工程套用模板的具体过程记录一下。 首先创建一个前端工程&#xff0c;我这里用的是Bootstrap5&#xff…

前端面试笔记vue

vue2 生命周期 beforeCreate&#xff1a;无data、methods、dom created&#xff1a;有data、methods&#xff0c;无dom beforeMount&#xff1a;有data&#xff0c;无dom mounted&#xff1a;有data&#xff0c;有dom beforeUpdate updated deforeDestroy destroy&#xff1a…