删除链表中重复的结点 -- Java实现

删除链表中重复的结点

https://www.nowcoder.com/questionTerminal/fc533c45b73a41b0b44ccba763f866ef?answerType=1&f=discussion

删除链表中重复的结点 -- Java实现

1. 辅助空间

1. 分析

多次遍历,第一次遍历把重复的结点值存入 set 容器,第二次遍历,当结点值存储在 set 容器中,就删除该结点

2. 代码

import java.util.*;
public class Solution {
    public ListNode deleteDuplication(ListNode pHead){
        if(pHead == null){
            return  null;
        }
        // 先找出相同结点,存入 set
        HashSet<Integer> set = new HashSet<>();
        ListNode pre = pHead;
        ListNode cur = pHead.next;
        while(cur != null){
            if(cur.val == pre.val){
                set.add(cur.val);
            }
            pre = cur;
            cur = cur.next;
        }
        // 再根据相同节点删除
        // 先删头部
        while(pHead != null && set.contains(pHead.val)){
            pHead = pHead.next;
        }
        if(pHead == null){
            return null;
        }
        // 再删中间结点
        pre = pHead;
        cur = pHead.next;
        while(cur != null){
            if(set.contains(cur.val)){
                pre.next = cur.next;
                cur = cur.next;
            }else{
                pre = cur;
                cur = cur.next;
            }
        }
        return pHead;
    }
}

3. 复杂度

时间复杂度:HashSet 是基于哈希表实现的,查找效率为 O(1),所以总的效率是 O(n)
空间复杂度:最坏的情况是存一半结点 O(n/2),最好的情况是一个也不存,O(1)

2. 遍历的同时删除

1. 分析

借助辅助头结点,可避免单独讨论头结点的情况。设置两个结点 pre 和 cur,当 cur 和 cur.next 值相等,cur 一直向前走,直到不等退出循环,这时候 cur 指的值还是重复值,调整 cur 和 pre 的指针再次判断

2. 代码

public class Solution {
    public ListNode deleteDuplication(ListNode pHead){
        if(pHead == null || pHead.next == null){
            return pHead;
        }
        // 自己构建辅助头结点
        ListNode head = new ListNode(Integer.MIN_VALUE);
        head.next = pHead;
        ListNode pre = head;
        ListNode cur = head.next;
        while(cur!=null){
            if(cur.next != null && cur.next.val == cur.val){
                // 相同结点一直前进
                while(cur.next != null && cur.next.val == cur.val){
                    cur = cur.next;
                }
                // 退出循环时,cur 指向重复值,也需要删除,而 cur.next 指向第一个不重复的值
                // cur 继续前进
                cur = cur.next;
                // pre 连接新结点
                pre.next = cur;
            }else{
                pre = cur;
                cur = cur.next;
            }
        }
        return head.next;
    }
}

3. 复杂度

时间复杂度:O(n)
空间复杂度:O(1)

全部评论
这题感觉不难,但是做了半天,头都晕了,服了。。哈哈
3 回复
分享
发布于 2020-01-17 13:53
辅助空间的思想真的很有用
1 回复
分享
发布于 2020-05-30 22:47
滴滴
校招火热招聘中
官网直投
删除时不使用 delete 关键字释放内存真的没关系么?
点赞 回复
分享
发布于 2019-09-18 14:01
是不是只能发现相邻重复的节点 不相邻的发现不了  比如1-2-3-1-2-3-1-2
点赞 回复
分享
发布于 2019-09-29 13:01
为什么不是报错?那个pre=cur,这里的pre都没有指向下一个节点,一直都是对同一个pre重新赋值
点赞 回复
分享
发布于 2019-12-25 17:57
照理说当pre=2 ,cur=3的时候经过循环cur变成4,然后pre.next=cur也会等于4,那结果不应该是1->2->4->5吗
点赞 回复
分享
发布于 2020-03-05 22:03
楼主,我想问一下虚拟头结点不能直接设为null吗,为什么会报空指针异常啊
点赞 回复
分享
发布于 2020-08-16 15:02
请问为什么要建立辅助头结点呢? ListNode head = new ListNode(Integer.MIN_VALUE); head.next = pHead; 这里没看懂,而且最后返回的是 head.next
点赞 回复
分享
发布于 2020-08-26 18:58
辅助头结点牛皮,想了好久,都没想到怎么处理头结点重复的解决办法
点赞 回复
分享
发布于 2021-03-06 00:39
太强了!
点赞 回复
分享
发布于 2021-04-12 14:14
想问下,输入[1,1,1]的情况下,最后pre.next==null了,但是head.next不是应该等于pHead吗,我看代码也一直没有对head进行操作,所以最后输出不应该是head.next=pHead吗,也就是[1,1,1]。
点赞 回复
分享
发布于 2021-06-08 12:48

相关推荐

投递腾讯云智研发等公司9个岗位
点赞 评论 收藏
转发
38 6 评论
分享
牛客网
牛客企业服务