• 代码随想录算法训练营第23期day3| 203.移除链表元素 ,707.设计链表,206.反转链表


    目录

    一、链表

    基础操作

    二、(leetcode 203)移除链表元素

    1.使用原来的链表

    2.设置虚拟头结点

    三、(leetcode 707)设计链表

    四、(leetcode 206)反转链表

    1.双指针法

    2.递归法


    一、链表

    单链表定义

    1. // 单链表
    2. struct ListNode {
    3. int val; // 节点上存储的元素
    4. ListNode *next; // 指向下一个节点的指针
    5. ListNode(int x) : val(x), next(NULL) {} // 节点的构造函数
    6. };

    基础操作

    1)删除节点

    2) 添加节点

    注意最后一个节点的删除,通过next指针进行删除操作 

    二、(leetcode 203)移除链表元素

    力扣题目链接

    状态:设置虚拟头结点方法AC,在原链表上直接操作待回顾

    思路:

    • 链表的其他节点是通过前一个节点来移除当前节点,头结点的移除是将头结点向后移动一位
    • 设置一个虚拟头结点,移除后return dummyNode->next

    1.使用原来的链表

    1. class Solution {
    2. public:
    3. ListNode* removeElements(ListNode* head, int val) {
    4. // 删除头结点
    5. while (head != NULL && head->val == val) { // 注意这里不是if,一串==val的
    6. ListNode* tmp = head;
    7. head = head->next;
    8. delete tmp;
    9. }
    10. // 删除非头结点
    11. ListNode* cur = head;
    12. while (cur != NULL && cur->next!= NULL) {
    13. if (cur->next->val == val) {
    14. ListNode* tmp = cur->next;
    15. cur->next = cur->next->next;
    16. delete tmp;
    17. } else {
    18. cur = cur->next;
    19. }
    20. }
    21. return head;
    22. }
    23. };

    2.设置虚拟头结点

    1. class Solution {
    2. public:
    3. ListNode* removeElements(ListNode* head, int val) {
    4. ListNode* dummyhead=new ListNode(0);//设置虚拟头节点
    5. dummyhead->next=head;
    6. ListNode* cur = dummyhead;
    7. while (cur->next != NULL) {
    8. if(cur->next->val == val) {
    9. ListNode* tmp = cur->next;
    10. cur->next = cur->next->next;
    11. delete tmp;
    12. } else {
    13. cur = cur->next;
    14. }
    15. }
    16. head = dummyhead->next;
    17. delete dummyhead;
    18. return head;
    19. }
    20. };

    三、(leetcode 707)设计链表

    力扣题目链接

    状态:其他功能已清楚,deleteAtIndex需要待看

    1. class MyLinkedList {
    2. public:
    3. // 定义链表节点结构体
    4. struct LinkedNode {
    5. int val;
    6. LinkedNode* next;
    7. LinkedNode(int val):val(val), next(nullptr){}
    8. };
    9. // 初始化链表
    10. MyLinkedList() {
    11. _dummyHead = new LinkedNode(0); // 这里定义的头结点 是一个虚拟头结点,而不是真正的链表头结点
    12. _size = 0;
    13. }
    14. // 获取到第index个节点数值,如果index是非法数值直接返回-1, 注意index是从0开始的,第0个节点就是头结点
    15. int get(int index) {
    16. if (index > (_size - 1) || index < 0) {
    17. return -1;
    18. }
    19. LinkedNode* cur = _dummyHead->next;
    20. while(index--){ // 如果--index 就会陷入死循环
    21. cur = cur->next;
    22. }
    23. return cur->val;
    24. }
    25. // 在链表最前面插入一个节点,插入完成后,新插入的节点为链表的新的头结点
    26. void addAtHead(int val) {
    27. LinkedNode* newNode = new LinkedNode(val);
    28. newNode->next = _dummyHead->next;
    29. _dummyHead->next = newNode;
    30. _size++;
    31. }
    32. // 在链表最后面添加一个节点
    33. void addAtTail(int val) {
    34. LinkedNode* newNode = new LinkedNode(val);
    35. LinkedNode* cur = _dummyHead;
    36. while(cur->next != nullptr){
    37. cur = cur->next;
    38. }
    39. cur->next = newNode;
    40. _size++;
    41. }
    42. // 在第index个节点之前插入一个新节点,例如index为0,那么新插入的节点为链表的新头节点。
    43. // 如果index 等于链表的长度,则说明是新插入的节点为链表的尾结点
    44. // 如果index大于链表的长度,则返回空
    45. // 如果index小于0,则在头部插入节点
    46. void addAtIndex(int index, int val) {
    47. if(index > _size) return;
    48. if(index < 0) index = 0;
    49. LinkedNode* newNode = new LinkedNode(val);
    50. LinkedNode* cur = _dummyHead;
    51. while(index--) {
    52. cur = cur->next;
    53. }
    54. newNode->next = cur->next;
    55. cur->next = newNode;
    56. _size++;
    57. }
    58. // 删除第index个节点,如果index 大于等于链表的长度,直接return,注意index是从0开始的
    59. void deleteAtIndex(int index) {
    60. if (index >= _size || index < 0) {
    61. return;
    62. }
    63. LinkedNode* cur = _dummyHead;
    64. while(index--) {
    65. cur = cur ->next;
    66. }
    67. LinkedNode* tmp = cur->next;
    68. cur->next = cur->next->next;
    69. delete tmp;
    70. //delete命令指示释放了tmp指针原本所指的那部分内存,
    71. //被delete后的指针tmp的值(地址)并非就是NULL,而是随机值。也就是被delete后,
    72. //如果不再加上一句tmp=nullptr,tmp会成为乱指的野指针
    73. //如果之后的程序不小心使用了tmp,会指向难以预想的内存空间
    74. tmp=nullptr;
    75. _size--;
    76. }
    77. // 打印链表
    78. void printLinkedList() {
    79. LinkedNode* cur = _dummyHead;
    80. while (cur->next != nullptr) {
    81. cout << cur->next->val << " ";
    82. cur = cur->next;
    83. }
    84. cout << endl;
    85. }
    86. private:
    87. int _size;
    88. LinkedNode* _dummyHead;
    89. };

    四、(leetcode 206)反转链表

    力扣题目链接

    状态:双指针法AC,递归法捋完思路AC、待回顾

    1.双指针法

    1. class Solution {
    2. public:
    3. ListNode* reverseList(ListNode* head) {
    4. ListNode* temp; // 保存cur的下一个节点
    5. ListNode* cur = head;
    6. ListNode* pre = NULL;
    7. while(cur) {
    8. temp = cur->next; // 保存一下 cur的下一个节点,因为接下来要改变cur->next
    9. cur->next = pre; // 翻转操作
    10. // 更新pre 和 cur指针
    11. pre = cur;
    12. cur = temp;
    13. }
    14. return pre;
    15. }
    16. };

    2.递归法

    递归解题三部曲:

    • 找整个递归的终止条件:递归应该在什么时候结束?
    • 找返回值:应该给上一级返回什么信息?
    • 本级递归应该做什么:在这一级递归中,应该完成什么任务?
    1. class Solution {
    2. public:
    3. ListNode* reverse(ListNode* pre,ListNode* cur){
    4. if(cur == NULL) return pre;
    5. ListNode* temp = cur->next;
    6. cur->next = pre;
    7. // 可以和双指针法的代码进行对比,如下递归的写法,其实就是做了这两步
    8. // pre = cur;
    9. // cur = temp;
    10. return reverse(cur,temp);
    11. }
    12. ListNode* reverseList(ListNode* head) {
    13. // 和双指针法初始化是一样的逻辑
    14. // ListNode* cur = head;
    15. // ListNode* pre = NULL;
    16. return reverse(NULL, head);
    17. }
    18. };

     

     

     

     

  • 相关阅读:
    【Redis】内存回收:内存淘汰策略
    arduino - 用 arduino zero 开发板来学习理解arduino软件编程细节
    Launcher app prediction
    基于价值的学习算法
    _linux 进程间通信(匿名管道)
    Java基础:简单工厂模式、工厂方法模式和抽象工厂模式综合概述
    android 12.0app应用卸载黑名单
    CSS 中背景background和img的区别和使用时机
    HTML知识点
    Spring-SpEL表达式超级详细使用全解
  • 原文地址:https://blog.csdn.net/weixin_42179093/article/details/133186437