鱼C论坛

 找回密码
 立即注册
查看: 1755|回复: 4

[已解决]这个数据结构问题困扰我很久了,有没有大佬清楚的解释一下。

[复制链接]
发表于 2023-4-26 13:20:53 | 显示全部楼层 |阅读模式
25鱼币
代码一:
  1. class ListNode {
  2.     int val;
  3.     ListNode next;

  4.     ListNode() {
  5.     }

  6.     ListNode(int val) {
  7.         this.val = val;
  8.     }

  9.     ListNode(int val, ListNode next) {
  10.         this.val = val;
  11.         this.next = next;
  12.     }
  13. }
  14. class Solution {
  15.     public ListNode removeElements(ListNode head, int val) {
  16.         ListNode dummyHead = new ListNode(0);
  17.         dummyHead.next = head;
  18.         ListNode temp = dummyHead;
  19.         while (temp.next != null) {
  20.             if (temp.next.val == val) {
  21.                 temp.next = temp.next.next;
  22.             } else {
  23.                 temp = temp.next;
  24.             }
  25.         }
  26.         return dummyHead.next;
  27.     }
  28. }
复制代码

代码二:
  1. class MyLinkedList {
  2.     int size;
  3.     ListNode head;

  4.     public MyLinkedList() {
  5.         size = 0;
  6.         head = new ListNode(0);
  7.     }

  8.     public int get(int index) {
  9.         if (index < 0 || index >= size) {
  10.             return -1;
  11.         }
  12.         ListNode cur = head;
  13.         for (int i = 0; i <= index; i++) {
  14.             cur = cur.next;
  15.         }
  16.         return cur.val;
  17.     }

  18.     public void addAtHead(int val) {
  19.         addAtIndex(0, val);
  20.     }

  21.     public void addAtTail(int val) {
  22.         addAtIndex(size, val);
  23.     }

  24.     public void addAtIndex(int index, int val) {
  25.         if (index > size) {
  26.             return;
  27.         }
  28.         index = Math.max(0, index);
  29.         size++;
  30.         ListNode pred = head;
  31.         for (int i = 0; i < index; i++) {
  32.             pred = pred.next;
  33.         }
  34.         ListNode toAdd = new ListNode(val);
  35.         toAdd.next = pred.next;
  36.         pred.next = toAdd;
  37.     }

  38.     public void deleteAtIndex(int index) {
  39.         if (index < 0 || index >= size) {
  40.             return;
  41.         }
  42.         size--;
  43.         ListNode pred = head;
  44.         for (int i = 0; i < index; i++) {
  45.             pred = pred.next;
  46.         }
  47.         pred.next = pred.next.next;
  48.     }
  49. }

  50. class ListNode {
  51.     int val;
  52.     ListNode next;

  53.     public ListNode(int val) {
  54.         this.val = val;
  55.     }
  56. }
复制代码


为什么代码二的第三行写了一句这个    ListNode head;  代码二的head.next才是真正的头节点,head却是一个虚拟节点。代码一却是head就是头节点。
哪位大佬能讲清楚这其中的区别吗
最佳答案
2023-4-26 13:20:54
hehexixi 发表于 2023-4-26 13:28
大佬,为什么ListNode head 就创建了一个虚拟头节点,其实我最想问的就是这个。

在代码二的 MyLinkedList 类中,我们看到在构造方法中创建了一个新的 ListNode,并将其赋值给 head:

  1. public MyLinkedList() {
  2.     size = 0;
  3.     head = new ListNode(0);
  4. }
复制代码


这里,我们创建了一个值为 0 的 ListNode 并将其赋值给 head。这个节点并不包含实际的数据,而只是作为一个占位符。因此,我们称它为虚拟头节点。

虚拟头节点的主要目的是简化链表操作。例如,在执行链表的插入或删除操作时,处理头部节点(第一个实际数据节点)通常需要特殊处理,因为它没有前驱节点。但是,如果我们有一个虚拟头节点,那么头部节点和其他节点就可以使用相同的逻辑来处理。

通过使用虚拟头节点,我们可以在链表的开头添加或删除节点,而不需要编写特殊的逻辑。虚拟头节点的 next 指针指向真正的头节点(即第一个实际数据节点)。由于虚拟头节点始终存在,我们可以确保 head.next 指向链表的第一个实际数据节点,即使链表为空。

所以,当我们说 ListNode head 创建了一个虚拟头节点时,我们实际上是说在 MyLinkedList 类的构造方法中,我们初始化了一个值为 0 的 ListNode,并将其作为虚拟头节点。这样做有助于简化链表操作和处理边界情况。

最佳答案

查看完整内容

在代码二的 MyLinkedList 类中,我们看到在构造方法中创建了一个新的 ListNode,并将其赋值给 head: 这里,我们创建了一个值为 0 的 ListNode 并将其赋值给 head。这个节点并不包含实际的数据,而只是作为一个占位符。因此,我们称它为虚拟头节点。 虚拟头节点的主要目的是简化链表操作。例如,在执行链表的插入或删除操作时,处理头部节点(第一个实际数据节点)通常需要特殊处理,因为它没有前驱节点。但是,如果我 ...
小甲鱼最新课程 -> https://ilovefishc.com
回复

使用道具 举报

发表于 2023-4-26 13:20:54 | 显示全部楼层    本楼为最佳答案   
hehexixi 发表于 2023-4-26 13:28
大佬,为什么ListNode head 就创建了一个虚拟头节点,其实我最想问的就是这个。

在代码二的 MyLinkedList 类中,我们看到在构造方法中创建了一个新的 ListNode,并将其赋值给 head:

  1. public MyLinkedList() {
  2.     size = 0;
  3.     head = new ListNode(0);
  4. }
复制代码


这里,我们创建了一个值为 0 的 ListNode 并将其赋值给 head。这个节点并不包含实际的数据,而只是作为一个占位符。因此,我们称它为虚拟头节点。

虚拟头节点的主要目的是简化链表操作。例如,在执行链表的插入或删除操作时,处理头部节点(第一个实际数据节点)通常需要特殊处理,因为它没有前驱节点。但是,如果我们有一个虚拟头节点,那么头部节点和其他节点就可以使用相同的逻辑来处理。

通过使用虚拟头节点,我们可以在链表的开头添加或删除节点,而不需要编写特殊的逻辑。虚拟头节点的 next 指针指向真正的头节点(即第一个实际数据节点)。由于虚拟头节点始终存在,我们可以确保 head.next 指向链表的第一个实际数据节点,即使链表为空。

所以,当我们说 ListNode head 创建了一个虚拟头节点时,我们实际上是说在 MyLinkedList 类的构造方法中,我们初始化了一个值为 0 的 ListNode,并将其作为虚拟头节点。这样做有助于简化链表操作和处理边界情况。
小甲鱼最新课程 -> https://ilovefishc.com
回复

使用道具 举报

发表于 2023-4-26 13:23:17 | 显示全部楼层
首先让我们分析一下这两段代码。代码一的目的是从给定的链表中移除特定值的元素。代码二实现了一个自定义链表类,提供了一些基本操作,如获取元素值、在头部添加元素、在尾部添加元素、在指定索引处添加元素以及删除指定索引处的元素。

现在,让我们解释你提到的区别。

在代码一中,我们移除链表中特定值的元素。head 是链表的第一个节点。然后,我们创建了一个虚拟头节点(dummyHead),并将其 next 指针指向原始链表的 head。这样做的原因是,在删除操作中,有可能需要删除原始链表的第一个节点,使用虚拟头节点可以简化代码,避免对链表头部的特殊处理。

在代码二中,我们实现了一个自定义链表类。类中的 ListNode head 是一个虚拟头节点。虚拟头节点的使用使得在链表头部插入或删除元素时不需要进行特殊处理,从而简化了代码。实际上,虚拟头节点不包含任何实际数据,它只是链表的一个占位符,方便我们操作链表。因此,真正的头节点是 head.next。

总之,代码一和代码二的主要区别在于:代码一中的 head 是原始链表的头节点,而代码二中的 head 是自定义链表类的虚拟头节点。虚拟头节点用于简化代码和处理边界情况,例如在链表头部插入或删除元素。
小甲鱼最新课程 -> https://ilovefishc.com
回复

使用道具 举报

 楼主| 发表于 2023-4-26 13:28:41 | 显示全部楼层
isdkz 发表于 2023-4-26 13:23
首先让我们分析一下这两段代码。代码一的目的是从给定的链表中移除特定值的元素。代码二实现了一个自定义链 ...

大佬,为什么ListNode head 就创建了一个虚拟头节点,其实我最想问的就是这个。
小甲鱼最新课程 -> https://ilovefishc.com
回复

使用道具 举报

 楼主| 发表于 2023-4-26 13:35:03 | 显示全部楼层
isdkz 发表于 2023-4-26 13:30
在代码二的 MyLinkedList 类中,我们看到在构造方法中创建了一个新的 ListNode,并将其赋值给 head:

...

清晰,真牛
小甲鱼最新课程 -> https://ilovefishc.com
回复

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Archiver|鱼C工作室 ( 粤ICP备18085999号-1 | 粤公网安备 44051102000585号)

GMT+8, 2025-4-26 20:43

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表