代码随想录算法训练营第33期day03: 第一章链表 part01

2024-03-09 23:12

本文主要是介绍代码随想录算法训练营第33期day03: 第一章链表 part01,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

203. 移除链表元素 10分18

        给你一个链表的头节点 head 和一个整数 val ,请你删除链表中所有满足 Node.val == val 的节点,并返回 新的头节点

示例 1:

输入:head = [1,2,6,3,4,5,6], val = 6

输出:[1,2,3,4,5]

示例 2:

输入:head = [], val = 1

输出:[]

示例 3:

输入:head = [7,7,7,7], val = 7

输出:[]


数据结构中的头结点和 leetCode 的头结点

        leetCode 中的默认的头结点是存储数据的,能可以为空,也参与题中所说的增删改查。但是,这与数据结构(严奶奶的)这本书所描述的数据结构并不相同,数据结构书上认为的链表的头结点一般是不存储数据的,若是要存储数据也一般是存储链表的长度,且头结点不参与链表的增删改查,头结点的作用在于统一链表的操作(空表和非空表),下面是 chatgpt 上的数据结构的说法。

        所以知道了上面的区别,这道题也就好做了,只要把头节点当作链表的第一个有数据的结点(首元结点)来看就行,就当作题目并没有给出头节点,因此我们需要设置一个虚拟的头结点,方便对链表的操作。


代码实现

ListNode* removeElements(ListNode* head, int val) {ListNode* vitualHead = new ListNode(0);                 // 设置虚拟头结点vitualHead->next = head;ListNode* p = vitualHead;while (p->next != NULL) {if (p->next->val == val) p->next = p->next->next;else p = p->next;}head = vitualHead->next;delete vitualHead;return head;
}

收获:发现了 leetCode 的头结点与我之前理解的头结点并不相同,以上也是链表的常规操作。


707. 设计链表 

        你可以选择使用单链表或者双链表,设计并实现自己的链表。单链表中的节点应该具备两个属性:val 和 next 。val 是当前节点的值,next 是指向下一个节点的指针/引用。

如果是双向链表,则还需要属性 prev 以指示链表中的上一个节点。假设链表中的所有节点下标从 0 开始。

实现 MyLinkedList 类:

  • MyLinkedList() 初始化 MyLinkedList 对象。
  • int get(int index) 获取链表中下标为 index 的节点的值。如果下标无效,则返回 -1 。
  • void addAtHead(int val) 将一个值为 val 的节点插入到链表中第一个元素之前。在插入完成后,新节点会成为链表的第一个节点。
  • void addAtTail(int val) 将一个值为 val 的节点追加到链表中作为链表的最后一个元素。
  • void addAtIndex(int index, int val) 将一个值为 val 的节点插入到链表中下标为 index 的节点之前。如果 index 等于链表的长度,那么该节点会被追加到链表的末尾。如果 index 比长度更大,该节点将 不会插入 到链表中。
  • void deleteAtIndex(int index) 如果下标有效,则删除链表中下标为 index 的节点。

示例:

输入

["MyLinkedList", "addAtHead", "addAtTail", "addAtIndex", "get", "deleteAtIndex", "get"]

[[], [1], [3], [1, 2], [1], [1], [1]]

输出: [null, null, null, null, 2, null, 3]

解释

MyLinkedList myLinkedList = new MyLinkedList();

myLinkedList.addAtHead(1);

myLinkedList.addAtTail(3);

myLinkedList.addAtIndex(1, 2); // 链表变为 1->2->3

myLinkedList.get(1); // 返回 2

myLinkedList.deleteAtIndex(1); // 现在,链表变为 1->3

myLinkedList.get(1); // 返回 3


注意事项+心得体会


        这道题怎么说呢,一开始不知道 leetCode 默认头结点也存储数据,写出来的代码怎么都 ac 不了,补来补去,后来看了卡哥的题解,才感觉有点奇怪,才发现原来是规定不一样,还有就是那个 index 真的不要理解成第几个,就是索引。还有一个就是链表的长度,我一开始是自己加了个函数获取链表长度,后来看了卡哥题解,觉得声明一个全局变量确实好用。这道题挺锻炼链表的掌握,就是链表的增删改查。下次还得刷!

class MyLinkedList {
public:ListNode* head;							// 创建空链表int _size;								// 链表长度MyLinkedList() {						// 初始化链表head = new ListNode(0);_size = 0;}/*获取链表中下标为 index 的节点的值。如果下标无效,则返回 -1 。*/int get(int index) {if (index < 0 || index + 1 > _size) return -1;ListNode* p = head->next;				// 索引 链表长度 = 索引+1while (index--) p = p->next;return p->val;}/*将一个值为 val 的节点插入到链表中第一个元素之前。在插入完成后,新节点会成为链表的第一个节点。*/void addAtHead(int val) {					// 头插法ListNode* node = new ListNode(val);node->next = head->next;head->next = node;_size++;								// 链表长度加一}/*将一个值为 val 的节点追加到链表中作为链表的最后一个元素。*/void addAtTail(int val) {					// 尾插法ListNode* p = head;while (p->next) p = p->next;			// 定位到链表尾端ListNode* node = new ListNode(val);node->next = p->next;p->next = node;_size++;}/*将一个值为 val 的节点插入到链表中下标为 index 的节点之前。如果 index 等于链表的长度,那么该节点会被追加到链表的末尾。如果 index 比长度更大,该节点将 不会插入 到链表中。*/void addAtIndex(int index, int val) {if (index > _size) return;				// 如果 index 比长度更大,该节点将 不会插入 到链表中。if (index < 0) index = 0;				// index < 0 默认插入到第一个ListNode* p = head;while (index--) p = p->next;ListNode* node = new ListNode(val);node->next = p->next;p->next = node;_size++;}/*如果下标有效,则删除链表中下标为 index 的节点。*/void deleteAtIndex(int index) {if (index < 0 || index >= _size) return;   // index 等于下标那就是越界ListNode* p = head;while (index--) p = p->next;ListNode* q = p->next;p->next = q->next;delete q;_size--;}
};

206. 反转链表 11分07

        给你单链表的头节点 head ,请你反转链表,并返回反转后的链表。

示例 1:

输入:head = [1,2,3,4,5]

输出:[5,4,3,2,1]

示例 2:

输入:head = [1,2]

输出:[2,1]

示例 3:

输入:head = []

输出:[]


解题思路+画图

这道题有两个思路

①简单思路(暴力)

        重新创建一个新的头节点 head_,遍历 head 将 head 链表的结点利用头插法的方法插入 head_,最后返回 head_

②遍历中交换结点


ListNode* reverseList(ListNode* head) {if (head == NULL) return head;ListNode* p = head;                     // 头结点也存储数据ListNode* cur = NULL;while (p) {                             // 不断移动结点-->画图ListNode* q = p->next;p->next = cur;cur = p;p = q;}return cur;  							// 根据画图得
}

收获:先画图,没有啥是一张图解决不了滴。

这篇关于代码随想录算法训练营第33期day03: 第一章链表 part01的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

深入理解Mysql OnlineDDL的算法

《深入理解MysqlOnlineDDL的算法》本文主要介绍了讲解MysqlOnlineDDL的算法,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小... 目录一、Online DDL 是什么?二、Online DDL 的三种主要算法2.1COPY(复制法)

Java集合之Iterator迭代器实现代码解析

《Java集合之Iterator迭代器实现代码解析》迭代器Iterator是Java集合框架中的一个核心接口,位于java.util包下,它定义了一种标准的元素访问机制,为各种集合类型提供了一种统一的... 目录一、什么是Iterator二、Iterator的核心方法三、基本使用示例四、Iterator的工

Java 线程池+分布式实现代码

《Java线程池+分布式实现代码》在Java开发中,池通过预先创建并管理一定数量的资源,避免频繁创建和销毁资源带来的性能开销,从而提高系统效率,:本文主要介绍Java线程池+分布式实现代码,需要... 目录1. 线程池1.1 自定义线程池实现1.1.1 线程池核心1.1.2 代码示例1.2 总结流程2. J

JS纯前端实现浏览器语音播报、朗读功能的完整代码

《JS纯前端实现浏览器语音播报、朗读功能的完整代码》在现代互联网的发展中,语音技术正逐渐成为改变用户体验的重要一环,下面:本文主要介绍JS纯前端实现浏览器语音播报、朗读功能的相关资料,文中通过代码... 目录一、朗读单条文本:① 语音自选参数,按钮控制语音:② 效果图:二、朗读多条文本:① 语音有默认值:②

Vue实现路由守卫的示例代码

《Vue实现路由守卫的示例代码》Vue路由守卫是控制页面导航的钩子函数,主要用于鉴权、数据预加载等场景,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着... 目录一、概念二、类型三、实战一、概念路由守卫(Navigation Guards)本质上就是 在路

uni-app小程序项目中实现前端图片压缩实现方式(附详细代码)

《uni-app小程序项目中实现前端图片压缩实现方式(附详细代码)》在uni-app开发中,文件上传和图片处理是很常见的需求,但也经常会遇到各种问题,下面:本文主要介绍uni-app小程序项目中实... 目录方式一:使用<canvas>实现图片压缩(推荐,兼容性好)示例代码(小程序平台):方式二:使用uni

JAVA实现Token自动续期机制的示例代码

《JAVA实现Token自动续期机制的示例代码》本文主要介绍了JAVA实现Token自动续期机制的示例代码,通过动态调整会话生命周期平衡安全性与用户体验,解决固定有效期Token带来的风险与不便,感兴... 目录1. 固定有效期Token的内在局限性2. 自动续期机制:兼顾安全与体验的解决方案3. 总结PS

C#中通过Response.Headers设置自定义参数的代码示例

《C#中通过Response.Headers设置自定义参数的代码示例》:本文主要介绍C#中通过Response.Headers设置自定义响应头的方法,涵盖基础添加、安全校验、生产实践及调试技巧,强... 目录一、基础设置方法1. 直接添加自定义头2. 批量设置模式二、高级配置技巧1. 安全校验机制2. 类型

Python屏幕抓取和录制的详细代码示例

《Python屏幕抓取和录制的详细代码示例》随着现代计算机性能的提高和网络速度的加快,越来越多的用户需要对他们的屏幕进行录制,:本文主要介绍Python屏幕抓取和录制的相关资料,需要的朋友可以参考... 目录一、常用 python 屏幕抓取库二、pyautogui 截屏示例三、mss 高性能截图四、Pill

使用MapStruct实现Java对象映射的示例代码

《使用MapStruct实现Java对象映射的示例代码》本文主要介绍了使用MapStruct实现Java对象映射的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,... 目录一、什么是 MapStruct?二、实战演练:三步集成 MapStruct第一步:添加 Mave