【一刷《剑指Offer》】面试题 13:在 O(1) 时间删除链表结点

2024-05-05 11:20

本文主要是介绍【一刷《剑指Offer》】面试题 13:在 O(1) 时间删除链表结点,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

力扣对应题目链接:LCR 136. 删除链表的节点 - 力扣(LeetCode)


一、《剑指 Offer》内容


二、分析题目

1、信息交换法

《剑指Offer》上给的这段代码,我把它称为信息交换法。
这道题其实考察了我们对链表的操作和时间复杂度的理解。

一般来讲,正常的解法时间复杂度都是 O(N),因为我们要找到待删除节点,不得不牺牲 O(N) 的时间复杂度。那么 O(1) 的时间复杂度是如何实现的呢?这里有一个信息交换法,即:假如 head 为 4 -> 5 -> 1 -> 9,val 为 5 -> 1 -> 9,表示我们要删除结点 5,这时我们使用信息交换,把 val 改为 1 -> 9的信息即可。为什么呢?因为我们把待删除节点的后一个元素赋值给待删除节点,也就相当于删除了当前元素。但也可以举出反例:如果 val 的值是最后一个元素 9 呢?我们无法找到 9 的后一个元素(因为没有),只能重头开始找待删除元素 9,这样的话时间复杂度再次变成了 O(N)。这就是这道题目有意思的地方,如果删除节点为前面的 n−1 个节点,时间复杂度均为 O(1),只有删除节点为最后一个的时候,时间复杂度才会变成 O(n)。平均时间复杂度为:(O(1)×(n−1)+O(n))/n=O(1),仍然为 O(1)。


【常规写法】

 2、单指针

要删除链表中节点值为 val 的节点,只需要进行如下两步操作:

  1. 找到待删除节点的前一节点 cur;
  2. 将 cur->next 设置为 cur->next->next。

注意:头节点可能是待删除的节点。 


3、双指针

  1. 设置两个指针分别指向头节点,pre (待删除节点的前一节点)和 cur (当前节点);
  2. 遍历整个链表,查找节点值为 val 的节点,找到了就删除该节点,否则继续查找。
  • 2.1. 找到,将当前节点的前驱节点(pre 节点或者说是之前最近一个值不等于 val 的节点)连接到当前节点(cur 节点)的下一个节点(pre->next = cur->next)。
  • 2.2. 没找到,继续遍历(cur = cur->next),更新最近一个值不等于 val 的节点(pre = cur)。

4、递归

递归的终止条件:

  1. 该节点为空节点,直接返回。

  2. 该节点就是要删除的节点,返回该节点的下一个节点。


5、设置虚拟(哨兵)头结点

之前的三种方法都需考虑头节点是否是待删除的节点,并且删除头节点的代码逻辑与删除非头节点的特别相似,可通过在头节点前面增加虚拟头节点,避免单独将拎头节点出来考虑,但是需要注意:返回的是虚拟头节点的下一节点而不是虚拟头节点。

注意:由于题目已明确是一个要删除的节点的值,因此找到该节点并删除之后,无需再遍历。


三、代码

1、参考代码

//更符合《剑指Offer》上题目的代码
class Solution {
public:ListNode* deleteNode(ListNode* head, ListNode* deleteNode) {if(head==nullptr || deleteNode==nullptr) return nullptr;// 要删除的节点不是尾节点if(deleteNode->next != nullptr){ListNode* nextNode = deleteNode->next;deleteNode->val = nextNode->val;deleteNode->next = nextNode->next;}// 链表只有一个节点,删除头节点(也是尾节点)else if(head == deleteNode){delete deleteNode;deleteNode = nullptr;head = nullptr;}// 链表中有多个节点,删除的节点是尾节点else{ListNode* p = head;//while(p->next != deleteNode) //另一种写法while(p->next != nullptr && p->next->next != nullptr){p = p->next;}p->next = nullptr;delete deleteNode;deleteNode = nullptr;}return head;}
};

2、对应题目代码

//单指针
/*** Definition for singly-linked list.* struct ListNode {*     int val;*     ListNode *next;*     ListNode(int x) : val(x), next(NULL) {}* };*/
class Solution {
public:ListNode* deleteNode(ListNode* head, int val) {if(head->val==val) return head->next;ListNode* cur=head;while(cur->next && cur->next->val!=val)cur=cur->next;if(cur->next)cur->next=cur->next->next;return head;}
};//双指针
/*** Definition for singly-linked list.* struct ListNode {*     int val;*     ListNode *next;*     ListNode(int x) : val(x), next(NULL) {}* };*/
class Solution {
public:ListNode* deleteNode(ListNode* head, int val) {if(head->val==val) return head->next;ListNode* prev=head;ListNode* cur=head;while(cur && cur->val!=val){prev=cur;cur=cur->next;}if(cur)prev->next=cur->next;return head;}
};//递归
/*** Definition for singly-linked list.* struct ListNode {*     int val;*     ListNode *next;*     ListNode(int x) : val(x), next(NULL) {}* };*/
class Solution {
public:ListNode* deleteNode(ListNode* head, int val) {if(head==NULL) return head;head->next=deleteNode(head->next, val);if(head->val==val) return head->next;else return head;}
};//设置虚拟头结点
/*** Definition for singly-linked list.* struct ListNode {*     int val;*     ListNode *next;*     ListNode(int x) : val(x), next(NULL) {}* };*/
class Solution {
public:ListNode* deleteNode(ListNode* head, int val) {ListNode* newHead=new ListNode(0);newHead->next=head;ListNode* cur=newHead;while(cur->next){if(cur->next->val==val){cur->next=cur->next->next;break;}else{cur=cur->next;}}return newHead->next;}
};

这篇关于【一刷《剑指Offer》】面试题 13:在 O(1) 时间删除链表结点的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/961598

相关文章

使用C++实现链表元素的反转

《使用C++实现链表元素的反转》反转链表是链表操作中一个经典的问题,也是面试中常见的考题,本文将从思路到实现一步步地讲解如何实现链表的反转,帮助初学者理解这一操作,我们将使用C++代码演示具体实现,同... 目录问题定义思路分析代码实现带头节点的链表代码讲解其他实现方式时间和空间复杂度分析总结问题定义给定

如何利用Java获取当天的开始和结束时间

《如何利用Java获取当天的开始和结束时间》:本文主要介绍如何使用Java8的LocalDate和LocalDateTime类获取指定日期的开始和结束时间,展示了如何通过这些类进行日期和时间的处... 目录前言1. Java日期时间API概述2. 获取当天的开始和结束时间代码解析运行结果3. 总结前言在J

docker如何删除悬空镜像

《docker如何删除悬空镜像》文章介绍了如何使用Docker命令删除悬空镜像,以提高服务器空间利用率,通过使用dockerimage命令结合filter和awk工具,可以过滤出没有Tag的镜像,并将... 目录docChina编程ker删除悬空镜像前言悬空镜像docker官方提供的方式自定义方式总结docker

修改若依框架Token的过期时间问题

《修改若依框架Token的过期时间问题》本文介绍了如何修改若依框架中Token的过期时间,通过修改`application.yml`文件中的配置来实现,默认单位为分钟,希望此经验对大家有所帮助,也欢迎... 目录修改若依框架Token的过期时间修改Token的过期时间关闭Token的过期时js间总结修改若依

Go Mongox轻松实现MongoDB的时间字段自动填充

《GoMongox轻松实现MongoDB的时间字段自动填充》这篇文章主要为大家详细介绍了Go语言如何使用mongox库,在插入和更新数据时自动填充时间字段,从而提升开发效率并减少重复代码,需要的可以... 目录前言时间字段填充规则Mongox 的安装使用 Mongox 进行插入操作使用 Mongox 进行更

使用Python在Excel中插入、修改、提取和删除超链接

《使用Python在Excel中插入、修改、提取和删除超链接》超链接是Excel中的常用功能,通过点击超链接可以快速跳转到外部网站、本地文件或工作表中的特定单元格,有效提升数据访问的效率和用户体验,这... 目录引言使用工具python在Excel中插入超链接Python修改Excel中的超链接Python

Android kotlin语言实现删除文件的解决方案

《Androidkotlin语言实现删除文件的解决方案》:本文主要介绍Androidkotlin语言实现删除文件的解决方案,在项目开发过程中,尤其是需要跨平台协作的项目,那么删除用户指定的文件的... 目录一、前言二、适用环境三、模板内容1.权限申请2.Activity中的模板一、前言在项目开发过程中,尤

对postgresql日期和时间的比较

《对postgresql日期和时间的比较》文章介绍了在数据库中处理日期和时间类型时的一些注意事项,包括如何将字符串转换为日期或时间类型,以及在比较时自动转换的情况,作者建议在使用数据库时,根据具体情况... 目录PostgreSQL日期和时间比较DB里保存到时分秒,需要和年月日比较db里存储date或者ti

C#实现添加/替换/提取或删除Excel中的图片

《C#实现添加/替换/提取或删除Excel中的图片》在Excel中插入与数据相关的图片,能将关键数据或信息以更直观的方式呈现出来,使文档更加美观,下面我们来看看如何在C#中实现添加/替换/提取或删除E... 在Excandroidel中插入与数据相关的图片,能将关键数据或信息以更直观的方式呈现出来,使文档更

Python脚本实现自动删除C盘临时文件夹

《Python脚本实现自动删除C盘临时文件夹》在日常使用电脑的过程中,临时文件夹往往会积累大量的无用数据,占用宝贵的磁盘空间,下面我们就来看看Python如何通过脚本实现自动删除C盘临时文件夹吧... 目录一、准备工作二、python脚本编写三、脚本解析四、运行脚本五、案例演示六、注意事项七、总结在日常使用