当前位置:   article > 正文

代码随想录 Day3 203.移除链表元素,707.设计链表,206.反转链表

代码随想录 Day3 203.移除链表元素,707.设计链表,206.反转链表

yi 移除链表元素

(1)不加虚拟头节点

本题目的核心思想通过对于数据结构知识的学习并不难,难在代码逻辑梳理将一个个零散的条件判断梳理为一个条件,并且记得c++需要主动释放内存。

  1. class Solution {
  2. public:
  3. ListNode* removeElements(ListNode* head, int val) {
  4. // 删除头结点
  5. while (head != NULL && head->val == val) { // 注意这里不是if
  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. };

问题:

1.不加虚拟头节点的时候,头节点的处理与非头节点的处理是不同的,但是尾节点不用单独分离出来处理。通过模拟删除非头节点可以发现cur是进行判断过的,需要进行判断的是cur->next,如果cur->next是nullptr则直接推出循环形成最终的子串。

2.不能对于空结点进行处理,所以对于块中需要处理的节点必须要保证是非空的,也就是while语句判断的一部分。

3.因此对于头节点的判断,条件:尝试以后发现需要的是头节点组合头节点值的判断

4.非头节点的处理,因为需要删除也就是跳过元素需要三个节点的位置,定位最开头的节点并对于cur和cur->next进行判空(类头节点不用对nextnext判断)。节点值的情况分为两种,一种直接删除,另一种继续向下。

5.删除的节点记得释放内存

(2) 采用虚拟头节点的方式

与上文非头节点一致

  1. class Solution {
  2. public:
  3. ListNode* removeElements(ListNode* head, int val) {
  4. ListNode* dummyHead = new ListNode(0); // 设置一个虚拟头结点
  5. dummyHead->next = head; // 将虚拟头结点指向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. };

问题:

1.new的格式:new ListNode(0);

er 707.设计链表

有比较多的细节需要注意

  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个节点之前插入一个新节点,例如index0,那么新插入的节点为链表的新头节点。
  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. };

问题:

1.首先一点就是要定义结构体,还有对于数据需要设置为private类型,需要设置两个数据分别是_size和虚结点。在操作之后不要忘记对于_size的值进行改变。

2.题目的描述:对于第i个元素进行操作,这个计数是从0开始的:第0个,第1个,第2个

3.不同的题目的return是不同的,注意不要进行简单复制。

return 常写成return

4.在delete操作中对于不需要的内存需要进行及时的释放,同时对于将这个已经释放的空间的指针设置为nullptr。

5.while循环中如果数值是0才是不满足条件,如果数值是正数或者是负数是满足条件的

如果while中判断的是一个指针的话,只有指针指向的是nullptr的时候才不满足。

san 206.反转链表

(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. };

问题:

1.了解这种从头到尾的遍历的方法,一开始的想到的是将前面的节点插入到尾部,但是超过时限。所以知道了这种将链表进行反转的方法。

(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. };

问题:

1.确定递归的出口,cur = nullptr;

确定每次传入改变的值,cur和pre

2.需要检查双指针法有哪些逻辑式子可以借助递归直接进行省略,还有一些则必须要列出来。练习时就少了cur->next = pre;

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/运维做开发/article/detail/920514
推荐阅读
相关标签
  

闽ICP备14008679号