Remove Duplicates from Sorted List

Tags: Linked List, Easy

Question

Problem Statement

Given a sorted linked list, delete all duplicates such that each element
appear only once.

For example,
Given 1->1->2, return 1->2.
Given 1->1->2->3->3, return 1->2->3.

题解

遍历之,遇到当前节点和下一节点的值相同时,删除下一节点,并将当前节点next值指向下一个节点的next, 当前节点首先保持不变,直到相邻节点的值不等时才移动到下一节点。

Python

  1. """
  2. Definition of ListNode
  3. class ListNode(object):
  4. def __init__(self, val, next=None):
  5. self.val = val
  6. self.next = next
  7. """
  8. class Solution:
  9. """
  10. @param head: A ListNode
  11. @return: A ListNode
  12. """
  13. def deleteDuplicates(self, head):
  14. curt = head
  15. while curt:
  16. while curt.next and curt.next.val == curt.val:
  17. curt.next = curt.next.next
  18. curt = curt.next
  19. return head

C++

  1. /**
  2. * Definition of ListNode
  3. * class ListNode {
  4. * public:
  5. * int val;
  6. * ListNode *next;
  7. * ListNode(int val) {
  8. * this->val = val;
  9. * this->next = NULL;
  10. * }
  11. * }
  12. */
  13. class Solution {
  14. public:
  15. /**
  16. * @param head: The first node of linked list.
  17. * @return: head node
  18. */
  19. ListNode *deleteDuplicates(ListNode *head) {
  20. ListNode *curr = head;
  21. while (curr != NULL) {
  22. while (curr->next != NULL && curr->val == curr->next->val) {
  23. ListNode *temp = curr->next;
  24. curr->next = curr->next->next;
  25. delete(temp);
  26. temp = NULL;
  27. }
  28. curr = curr->next;
  29. }
  30. return head;
  31. }
  32. };

Java

  1. /**
  2. * Definition for ListNode
  3. * public class ListNode {
  4. * int val;
  5. * ListNode next;
  6. * ListNode(int x) {
  7. * val = x;
  8. * next = null;
  9. * }
  10. * }
  11. */
  12. public class Solution {
  13. /**
  14. * @param ListNode head is the head of the linked list
  15. * @return: ListNode head of linked list
  16. */
  17. public static ListNode deleteDuplicates(ListNode head) {
  18. ListNode curr = head;
  19. while (curr != null) {
  20. while (curr.next != null && curr.val == curr.next.val) {
  21. curr.next = curr.next.next;
  22. }
  23. curr = curr.next;
  24. }
  25. return head;
  26. }
  27. }

源码分析

  1. 首先进行异常处理,判断head是否为NULL
  2. 遍历链表,curr->val == curr->next->val时,保存curr->next,便于后面释放内存(非C/C++无需手动管理内存)
  3. 不相等时移动当前节点至下一节点,注意这个步骤必须包含在else中,否则逻辑较为复杂

while 循环处也可使用curr != null && curr.next != null, 这样就不用单独判断head 是否为空了,但是这样会降低遍历的效率,因为需要判断两处。使用双重while循环可只在内循环处判断,避免了冗余的判断,谢谢 @xuewei4d 提供的思路。

复杂度分析

遍历链表一次,时间复杂度为 O(n), 使用了一个中间变量进行遍历,空间复杂度为 O(1).

Reference