Java 数据结构之删除链表中重复的结点
作者:2021dragon 发布时间:2023-11-28 15:36:22
标签:Java,删除,链表,结点
核心考点:链表操作,临界条件检查,特殊情况处理
在一个排序的链表中,存在重复的结点,请删除该链表中重复的结点,重复的结点不保留,返回链表头指针。
解析一:(不提倡)
解决该问题较简单,且在写代码时不易出错的做法如下:
遍历一遍链表,记录重复结点的结点值。
再遍历一遍链表,逐个删除重复结点。
动图演示:
该方法需要遍历两遍链表,且需要开辟额外的内存空间存储重复结点的结点值,所以一般不提倡。
/*
struct ListNode {
int val;
struct ListNode *next;
ListNode(int x) :
val(x), next(NULL) {
}
};*/
class Solution {
public:
ListNode* deleteDuplication(ListNode* pHead) {
if(pHead == nullptr||pHead->next == nullptr) //链表为空或只有一个结点无需进行操作
return pHead;
ListNode* cur = pHead;
unordered_set<int> filter;
//1、遍历链表找出重复的结点,将结点值存入filter
while (cur->next)
{
if (cur->val == cur->next->val)
filter.insert(cur->val);
cur = cur->next;
}
//2、逐个删除重复的结点
//先删除头部的重复结点
while(pHead&&filter.find(pHead->val) != filter.end())
{
pHead = pHead->next;
}
if(pHead == nullptr)
return nullptr;
//再删除其余的重复结点
ListNode* prev = pHead;
ListNode* last = prev->next;
while(last)
{
if(filter.find(last->val) != filter.end())
{
prev->next = last->next;
last = last->next;
}
else
{
prev = prev->next;
last = last->next;
}
}
return pHead; //返回链表头指针
}
};
解析二:(正解)
我们当然应该尽可能在遍历一遍链表的情况下解决该问题,这时我们需要使用两个指针配合完成,该过程当中包含大量细节,大致步骤如下:
1.为了后续操作方便,先为该链表创建一个头结点。
2.使用指针prev和last遍历链表,初始时prev指向头结点,last指向头结点的下一个结点。
3.当last指向的结点值与其后一个结点的结点值相同时,last独自后移,直到last指向结点的结点值与其下一个结点的结点值不同为止,此时让prev指向的结点指向last的后一个结点,最后让last指向下一个结点(图中未后移)。
4.当last指向的结点值与其后一个结点的结点值不同时,prev和last一同向后移。
如此进行下去,直到last将链表遍历完,链表当中重复的结点也就全部被删除了,最后返回头结点指向的链表即可。
动图演示:
/*
struct ListNode {
int val;
struct ListNode *next;
ListNode(int x) :
val(x), next(NULL) {
}
};*/
class Solution {
public:
ListNode* deleteDuplication(ListNode* pHead) {
if (pHead == nullptr || pHead->next == nullptr) //链表为空或只有一个结点无需进行操作
return pHead;
ListNode* head = new ListNode(0); //创建头结点(便于操作)
head->next = pHead; //头结点与原链表建立关系
ListNode* prev = head;
ListNode* last = prev->next;
while (last)
{
//未发现重复的结点,prev和last一同后移
while (last->next&&last->val != last->next->val)
{
prev = prev->next;
last = last->next;
}
//发现重复的结点,last独自后移
while (last->next&&last->val == last->next->val)
{
last = last->next;
}
//到达此处有三种情况:
//1、没有需要删除的重复结点,是因为last->next == nullptr到此
//2、有需要删除的重复结点,是因为last->next == nullptr到此(链表后半段都需要删除)
//3、有需要删除的重复结点,是因为last->val != last->next->val到此(链表中间某段需要删除)
if (prev->next != last) //说明有需要删除的重复结点
{
prev->next = last->next;
}
last = last->next;
}
return head->next; //返回链表头指针
}
};
来源:https://blog.csdn.net/chenlong_cxy/article/details/120649781


猜你喜欢
- Ping pingSender = new Ping(); PingReply reply = pingSender.Send("
- 本文实例为大家分享了Spring boot实现文件上传的具体代码,供大家参考,具体内容如下1. 创建一个Maven的web工程,然后配置po
- 前言最近在学习Spring Boot结合Redis时看了一些网上的教程,发现这些教程要么比较老,要么不知道从哪抄得,运行起来有问题。这里分享
- 本文实例为大家分享了java实现斗地主发牌的具体代码,供大家参考,具体内容如下参考斗地主的游戏规则,完成一个发牌的功能(54张牌,考虑点数,
- 1. HTTP 响应码 301 和 302 代表的是什么?有什么区别?301:永久重定向。 302:暂时重定向。它们的区别是,301 对搜索
- 前言以键值对Dictionary<[key], [value]>形式存值,和哈希表很像也是一种无序的结构。要使用Dictiona
- 双向循环链表定义相比于单链表,有两个指针,next指针指向下一个结点,prior指针指向上一个结点,最后一个结点的next指针指向头结点,头
- 初学线程时,总是将 run 方法和 start 方法搞混,虽然二者是完全不同的两个方法,但刚开始使用时很难分清,原因就是因为初次使用时效果貌
- 1.XxlJob简介官方网址:https://www.xuxueli.com/xxl-jobXXL-JOB是一个分布式任务调度平台,其核心设
- 本文实例为大家分享了Android自动播放Banner图片轮播的具体代码,供大家参考,具体内容如下先看一下效果图支持本地图片以及网络图片or
- 这篇文章主要介绍了JAVA利用递归删除文件代码实例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可
- 前言.NET 生态越来越好,初学的朋友也越来越多。处理同一件简单的问题,随着我们知识的积累解决问题的方法也会越来越多。开始学习一门新的语言,
- 近日工程中,逐渐感觉到原来复制粘贴代码的笨重,突然想起以前有人和我说起过Git和SVN之类的版本管理工具。由于平时主要是写Java代码,所以
- 目录Shiro简介Shiro快速入门SpringBoot-Shiro整合(最后会附上完整代码)附上最后的完整代码Shiro整合mybatis
- 前言在最近的一个项目需要用JAVA来解析DICOM图片,DICOM被广泛应用于放射医疗,心血管成像以及放射诊疗诊断设备(X射线,CT,核磁共
- XSS是一种经常出现在web应用中的计算机安全漏洞,具体信息请自行Google。本文只分享在Spring Cloud Gateway中执行通
- 本文实例为大家分享了Android简单实现天气预报App的具体代码,供大家参考,具体内容如下一、UI设计首页UI<?xml versi
- 本文介绍了详解Maven * Nexus的安装与使用,分享给大家,具体如下:1.安装1.1 安装docker并加速yum update &am
- 用途:IO工具类(将内容写到流中)使用场景IO工具类只是辅助流的读写,并不负责关闭流。原因是流可能被多次读写,读写关闭后容易造成问题。项目引
- Unity IPostprocessBuild技术文章Unity IPostprocessBuild是Unity引擎中的一个非常有用的功能,