链表之删除单链表中的重复节点

news2025/1/22 21:55:49

文章目录

      • 删除单链表中的重复节点
        • 题目描述
        • 解题思路
        • 代码实现

删除单链表中的重复节点

力扣链接

题目描述

编写代码,移除未排序链表中的重复节点。保留最开始出现的节点。

 示例1:
 输入:[1, 2, 3, 3, 2, 1]
 输出:[1, 2, 3]
 示例2:
 输入:[1, 1, 1, 1, 2]
 输出:[1, 2]

提示:

  1. 链表长度在[0, 20000]范围内。
  2. 链表元素在[0, 20000]范围内。

解题思路

思路一:

  • 定义两个指针current和p来逐个遍历链表
    • current指向的元素依次与p指向的元素的下一个元素进行比较
      • 若相同则删去p的下一个元素【即使p的next指向下下个元素(p的next的next),继续循环比较】
      • 若不相同则p往后移动一个位置,继续循环比较,直到p所指元素的下一个为空【只要后一个不为空则继续比较,否则 停止本次比较】
    • 若所有元素都与后面元素比较过了,则返回当前链表head

思路二:

  • 定义p指针遍历链表,利用set,遇到一个元素就去set中找是否包含这个元素
    • 如果set中包含了当前元素,就说明当前遍历的元素属于重复节点,则删除
      • 利用s.count查询函数,返回的是一个整数,表示查询到元素的个数
    • 如果不包含,则添加进set中,并将p指针后移

思路三:利用标记数组,标记当前值是否出现过

代码实现

//提供完整的代码,可以在C编辑器中进行测试

#include <iostream>
#include <set>

using namespace std;

struct ListNode{
	int val;
	ListNode *next;
	ListNode(int x) : val(x),next(NULL){}
}; 

class Solution{
public:
	//方法一: 
	ListNode* removeDuplicateNodes(ListNode* head){
		//判断是否需要遍历
        if(head==NULL||head->next==NULL) return head;
		
		ListNode* current = head;
		while(current){//第一重循环:将第i个元素依次与后面元素进行比较(i从1到最后)
			ListNode *p = current;
			while(p->next){//第二重循环:与后1个比、后2个比、后三个比...与最后一个比
				if(p->next->val == current->val){
					p->next = p->next->next;//删除p后面这个重复的元素
				}else{
					p = p->next;//p后移
				}
			}
			current = current->next;//将第i+1个元素开始依次与后面元素进行比较
		}
		return head;//这里返回的一定是head,此时的head后面的链表已经删除了重复的元素 
	}

	//方法二: 
	ListNode* removeDuplicateNodes2(ListNode* head){
		if(head==NULL||head->next==NULL) return head;
		
		set<int> s;
		
		ListNode* p = head;
		s.insert(p->val);
		
		while(p->next){//只要还没到最后一个元素,就遍历
			if(s.count(p->next->val)!=0){//包含了,则重复
				p->next = p->next->next; 
			}else{						//不重复,添加进去、后移
				s.insert(p->next->val);	//或者可以先后移再添加,p=p->next;s.insert(p->val);
				p = p->next;
			}
		}
		return head;
	}
    
    //方法三:
	 ListNode* removeDuplicateNodes3(ListNode* head){
	 	if(head==NULL||head->next==NULL) return head;
	 	
	 	//利用val的值在0~20000之间,标记是否出现过 
	 	int index[20001] = {0};
	 	index[head->val] = 1;
	 	
	 	struct ListNode* q = head;
	 	
	 	while(q->next){
	 		//下一个val未出现过,保存下一个val 
	 		if(index[q->next->val]==0){
	 			index[q->next->val] = 1;
	 			q = q->next;
			 }else{							//下一个val出现过,删除下一个节点 
			 	q->next = q->next->next;
			 }
		 }
		 return head;
	 }
	
    //链表打印函数
	void ListPrintf(ListNode* head){
		
		ListNode* pos = head;
	 
		while(pos){
			cout<<pos->val<<"->";
			pos = pos->next; 
		}
		cout <<"NULL"<<endl;	
	}
	
    //尾部插入函数
	void ListBackInsert(ListNode** pphead,int val){
		ListNode* newNode = new ListNode(val);
		
		if(*pphead==NULL){
			*pphead = newNode;
		}else{
			ListNode* pos = *pphead;
			
			while(pos->next!=NULL){
				pos = pos->next;
			}
			pos->next = newNode;
		}
	} 
};


int main(){
	ListNode* test = new ListNode(1);
	ListNode* result;
	
	class Solution s;	
	
	s.ListBackInsert(&test,2);
	s.ListBackInsert(&test,3);
	s.ListBackInsert(&test,3);
	s.ListBackInsert(&test,2);
	s.ListBackInsert(&test,1);
	s.ListPrintf(test);
	
	result = s.removeDuplicateNodes(test);
	//result = s.removeDuplicateNodes2(test);
	//result = s.removeDuplicateNodes3(test);
	s.ListPrintf(result);

	return 0;
}

结果:
image-20221127180021258

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

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

相关文章

WEB网站安全检测系统设计与实现

目 录 1 引言 1 2 Web服务器所受的威胁及防御 1 2.1 缓冲区溢出 1 2.2 SQL注入攻击 1 2.3 基于脚本的DDos攻击 2 2.4 其他的不安全因素 3 3 Web的木马检测系统的设计 4 3.1 体系结构 4 3.2 处理流程 5 3.3 对客户端访问的响应 7 3.4 策略引擎的设计 8 3.4.1 策略的属性 8 3.4.2…

9、软件包管理

文章目录9、软件包管理9.1 RPM9.1.1 RPM 概述9.1.2 RPM 查询命令&#xff08;rpm -qa&#xff09;9.1.3 RPM 卸载命令&#xff08;rpm -e&#xff09;9.1.4 RPM 安装命令&#xff08;rpm -ivh&#xff09;9.2 YUM 仓库配置9.2.1 YUM 概述9.2.2 YUM 的常用命令9.2.3 修改网络 YU…

Linux系统中基本的启动方式

大家好&#xff0c; 今天主要和大家聊一聊&#xff0c;Linux系统的启动方式有哪些&#xff1f; 目录 第一&#xff1a;启动方式基本简介 第二​&#xff1a;启动模式的选择 第一&#xff1a;启动方式基本简介 Linux系统支持多种启动方式&#xff0c;可以从SD/EMMC、NAND Flas…

Jumperserver堡垒机管理服务器实战

一、 Jumpserver堡垒机简介 1、跳板机简介 跳板机就是一台服务器,开发或运维人员在维护过程中首先要统一登录到这台服务器,然后再登录到目标设备进行维护和操作。 跳板机缺点:没有实现对运维人员操作行为的控制和审计,使用跳板机的过程中还是会出现误操作、违规操作导致的…

Vue实现角色权限动态路由详细教程,在vue-admin-template基础上修改,附免费完整项目代码

前言 vue-admin-template是一个最基础的后台管理模板&#xff0c;只包含了一个后台需要最基础的东西&#xff0c;如果clone的是它的master分支&#xff0c;是没有权限管理的&#xff0c;只有完整版vue-element-admin有这个功能&#xff0c;但是为了小小的一个权限管理而用比较…

java框架 Spring之 AOP 面向切面编程 切入点表达式 AOP通知类型 Spring事务

AOP(Aspect Oriented Programming)面向切面编程&#xff0c;一种编程范式&#xff0c;指导开发者如何组织程序结构 作用&#xff1a;在不惊动原始设计的基础上为其进行功能增强 Spring理念&#xff1a;无入侵式/无侵入式 我们在不修改源代码的时候&#xff0c;为了执行另外的…

idea如何快速找到项目中对应的类(包括源码)

文章目录1. 前言2. 先说结论3. idea的全局搜索功能 MethodValidation4. 搜索spring源码(例子)1. 前言 最近在看某些功能的时候&#xff0c;会去看对应的源码&#xff0c;而有时候只知道类名&#xff0c;不知道从哪里进入源码&#xff0c;因此就比较好奇&#xff0c;idea的全局…

JAVA类加载器

JAVA是一种解释型语言&#xff0c;也就是一种边解释边执行的语言。JAVA所有源代码在执行之前&#xff0c;先要被编译成class文件&#xff0c;然后类加载器加载解析class文件&#xff0c;最后才执行。 JVM自带了几个类型的类加载器&#xff0c;JVM使用分层的软件架构方式设计类…

蜂鸟E203学习笔记(二)--蜂鸟E203总体框架

蜂鸟E203总体框架 蜂鸟E203处理器系统如下图所示 一、蜂鸟E203处理器核设计总览和顶层 1.1 蜂鸟E203处理器核的设计理念 模块化和可重用性&#xff1a;将处理器分成几个主体模块&#xff0c;每个单元之间的接口简单清晰。面积最小化&#xff1a;追求低功耗和小面积&#x…

DS18B20

一、DS18B20初始化时序图 &#xff08;1&#xff09; 先将数据线置高电平“1”。 &#xff08;2&#xff09; 延时&#xff08;该时间要求的不是很严格&#xff0c;但是尽可能的短一点&#xff09;。 &#xff08;3&#xff09; 数据线拉到低电平“0”。 &#xff08;4&#xf…

【配送路径规划】基于matlab遗传算法求解静态外卖骑手路径规划问题【含Matlab源码 2248期】

⛄一、遗传算法求解静态外卖骑手路径规划问题 1 模型假设 外卖配送的实际运行是一个复杂的过程, 受诸多因素影响, 为了建立调度模型, 本文做如下假设。 (1) 外卖配送更多的是服务特殊群体, 所以本文认为外卖配送是一种预约型配送, 即在进行调度安排前, 己经获取了所有顾客的地…

Java并发-生产者消费者实现

生产者与消费者模型介绍 定义&#xff1a; 生产者消费者模式是一个十分经典的多线程并发协作的模式。 意义&#xff1a;弄懂生产者消费者问题能够让我们对并发编程的理解加深。 介绍&#xff1a;所谓生产者 - 消费者问题&#xff0c;实际上主要是包含了两类线程&#xff0c;…

Python面向对象编程之对象行为与特殊方法

面向对象编程之对象行为与特殊方法 python中的对象通常根据它们的行为和实现的功能进行分类。例如&#xff0c;所有序列类型都分在一组&#xff0c;如字符串&#xff0c;列表和元组&#xff0c;就是因为它们都支持一组相同的序列操作&#xff0c;如s[n], len[s]等。 所有基本…

【学习总结】LSD-SLAM配置与运行记录

今天安装测试了LSD-SLAM&#xff0c;记录配置中遇到的问题。 LSD-SLAM论文 LSD-SLAM: Large-Scale Direct Monocular SLAM, J. Engel, T. Schps, D. Cremers, ECCV 14 Semi-Dense Visual Odometry for a Monocular Camera, J. Engel, J. Sturm, D. Cremers, ICCV 13 配置环境…

SpringBoot实现多数据源(四)【集成多个 Mybatis 框架】

上一篇文章《SpringBoot实现多数据源&#xff08;三&#xff09;【AOP 自定义注解】》 四、集成多个 Mybatis 框架 实现步骤 创建一个 dynamic_mybatis 的springboot项目&#xff0c;导入依赖 pom.xml <dependencies><!--jdbc--><dependency><groupId…

Fiddler抓取手机app包

文章目录1. 配置fiddler1.1 下载fiddler1.2 配置fiddler2. 安装证书2.1 查询主机ip2.2下载证书2.3 手机安装证书2.4 查询安装的证书3. 手机设置代理4. 测试是否抓包成功大前提&#xff1a;手机和Fiddler所在的主机在同一网段&#xff0c;且能够互相访问 1. 配置fiddler 1.1 下…

第三十八篇 Vue中封装Swiper组件 2.0

上一篇内容讲到封装Swiper组件的一个过程&#xff0c;如果是静态的数据封装组件初始化在mounted当中并无多大影响&#xff0c;但是这样封装的组件复用性较低或者可能只使用一次&#xff0c;那么在动态使用通过ajax请求数据需要面临的是swiper初始化过早的问题&#xff0c;在mou…

【车间调度】基于matlab混合蛙跳算法 (SFLA) 求解简单调度问题【含Matlab源码 2247期】

⛄一、车间调度简介 在传统的SFLA中&#xff0c;每一个青蛙的位置代表一个解&#xff0c;若干个青蛙组成的种群代表一个解的集合&#xff0c;种群被划分为不同的组&#xff0c;即模因组&#xff0c;对每个模因组执行搜索过程&#xff0c;当达到终止条件后&#xff0c;重新将模…

编译原理13:SLR(1)分析表、LR(1)分析表

更强的LR分析 可以根据当前单词&#xff0c;来选择是移进还是归约。只要所有移进项目中的点后面的那些终结符&#xff0c;与归约项目生成的非终结符的Follow集合的元素没有重叠。若当前单词属于上述Follow集合里则规约 SLR(1)冲突解决办法 SLR(1)分析表的构造 SLR(1)分析表的构…

001. 组合

1.题目链接&#xff1a; 77. 组合 2.大概思路&#xff1a; 2.1题目要求&#xff1a; 给两个值 n 和 k &#xff0c;要求从[1&#xff0c;n]的区间中&#xff0c;输出所有元素数量为k的组合。&#xff08;不能有[1,1]&#xff0c;值只能取一次&#xff09; 2.2思路&#xff…