谷歌面试题:如何从无序链表中移除重复项?有几种方式?

一位小伙伴来问一道谷歌的笔试题,关于单链表操作的,问到底有多少种解决方案,今天我们就来聊聊。

题目的大致意思是:
假设存在一个无序单链表,将重复结点去除后,并保原顺序。
去重前:1→3→1→5→5→7
去重后:1→3→5→7

顺序删除

通过双重循环直接在链表上执行删除操作。外层循环用一个指针从第一个结点开始遍历整个链表,然后内层循环用另外一个指针遍历其余结点,将与外层循环遍历到的指针所指结点的数据域相同的结点删除,如下图所示。

假设外层循环从outerCur开始遍历,当内层循环指针innerCur遍历到上图实线所示的位置(outerCur.data==innerCur.data)时,此时需要把innerCur指向的结点删除。

具体步骤如下:

  • 用tmp记录待删除的结点的地址。
  • 为了能够在删除tmp结点后继续遍历链表中其余的结点,使innerCur指针指向它的后继结点:innerCur=innerCur.next
  • 从链表中删除tmp结点。

实现代码如下:

运行结果:

算法性能分析

由于这种方法采用双重循环对链表进行遍历,因此,时间复杂度为O(N^2)。其中,N为链表的长度。在遍历链表的过程中,使用了常量个额外的指针变量来保存当前遍历的结点、前驱结点和被删除的结点,因此,空间复杂度为O(1)

递归法

主要思路为:对于结点cur,首先递归地删除以cur.next为首的子链表中重复的结点,接着从以cur.next为首的子链表中找出与cur有着相同数据域的结点并删除。

实现代码如下:

算法性能分析

这种方法与方法一类似,从本质上而言,由于这种方法需要对链表进行双重遍历,因此,时间复杂度为O(N^2)。其中,N为链表的长度。由于递归法会增加许多额外的函数调用,因此,从理论上讲,该方法效率比前面的方法低。

空间换时间

通常情况下,为了降低时间复杂度,往往在条件允许的情况下,通过使用辅助空间实现。

具体而言,主要思路如下。

  • 建立一个HashSet,HashSet中的内容为已经遍历过的结点内容,并将其初始化为空。

  • 从头开始遍历链表中的所以结点,存在以下两种可能性:

    • 如果结点内容已经在HashSet中,则删除此结点,继续向后遍历。

    • 如果结点内容不在HashSet中,则保留此结点,将此结点内容添加到HashSet中,继续向后遍历。

「引申:如何从有序链表中移除重复项?」

如链表:1,3、5、5、7、7、8、9

去重后:1,3、5、7、8、9

分析与解答

上述介绍的方法也适用于链表有序的情况,但是由于以上方法没有充分利用到链表有序这个条件,因此,算法的性能肯定不是最优的。本题中,由于链表具有有序性,因此,不需要对链表进行两次遍历。所以,有如下思路:用cur 指向链表第一个结点,此时需要分为以下两种情况讨论。

  • 如果cur.data==cur.next.data,那么删除cur.next结点。

  • 如果cur.data!=cur.next.data,那么cur=cur.next,继续遍历其余结点。

总结

对于无序单链表中,想要删除其中重复的结点(多个重复结点保留一个)。删除办法有按照顺序删除、使用递归方式删除以及可以使用空间换时间(HashSet中元素的唯一性)。

最近有读者想要分布式的项目,还有想要商城的,还有想要springboot,springcloud,k8s等等,这次直接分享几乎涵盖了我们java程序员的大部分技术桟,可以说真的非常全面了。强烈建议大家都上手做一做,而且以后肯定用的上。

资料包含高清视频+课件+源码……

扫以下二维码并回复“1002”即可获取

点赞越多,bug越少

(0)

相关推荐

  • 【漫画】什么是二叉树?

    本文参考河北小博的文章 https://blog.csdn.net/qq_32799165/article/details/88879239 漫画由小猿编写创作 很多新手可能刚接触到数据结构,下面先给 ...

  • 龙门阵 | 在婚姻中你们之间是用哪一种方式进行谈判的?

    你是哪一种谈判者?你的伴侣又是哪一种谈判者?顺从型的?还是霸道型的?甚至是狡诈型的?这对你们的夫妻关系是至关重要的. 答案在这次伉俪疗法中会显示出来的,这个疗法在很大程度上还能反映出你跟伴侣共处的协调 ...

  • 反向保理供应链金融ABS业务中,避免过桥资金的三种方式!

    即在专项计划设立前,保理商以商业承兑汇票(以下简称"商票")向供应商支付基础资产对应应收账款转让价款,商票的承兑日安排在专项计划设立日以后. 专项计划设立后,保理商以专项计划支付的 ...

  • 【Excel】获取一列中的不重复值,四种方法都有了

    获取不重复值的方法有很多,例如高级筛选法.透视表法.基础操作法和公式法.本例分别向大家介绍这四种方法如何使用. 高级筛选法获取不重复值: 首先,选中A列的数据区域,选择[数据]-[筛选]-[高级]. ...

  • 如何删除 Excel 表格中的所有重复行?4 种方法都很简便

    如果数据表的某一列中有重复单元格,要去重还是比较容易的,但是如果数据表中存在所有单元格完全重复的行,如何快速找到这些重复行并且去重呢? 案例: 下图中的数据表分别有两对完全重复的行,请删除所有重复行. ...

  • Excel工作表中,删除重复数据的2种方法解读,高效且实用!

    在实际的工作中,经常要对工作表中重复的数据进行删除,如若数据行只有几条,则可通过人工查找出来,如若数据行较多,这种方法就费时费力,而且容易出错,造成对表格数据的准确性和个人能力的怀疑--今天,小编给大 ...

  • 聊聊业务系统中投递消息到mq的几种方式

    背景 电商中有这样的一个场景: 下单成功之后送积分的操作,我们使用mq来实现 下单成功之后,投递一条消息到mq,积分系统消费消息,给用户增加积分 我们主要讨论一下,下单及投递消息到mq的操作,如何实现 ...

  • Excel中提取不重复值的 5 种操作,还不会的可以拿去开练了!

    表格中存在重复数值是我们经常遇到的一个问题,今天技巧妹整理了提取不重复值的5种常用操作,还不会的赶紧拿去开练. 1.删除重复值 利用[数据]选项卡下的"删除重复项"这一功能保留唯一 ...

  • 在生活的对境中,训练自我觉察的15种方式

    对境,是检验心性的最佳考场 当遇见了某种情景,某个人, 能否及时反观自身, 与涌起的情绪和冲动做一次深度沟通? 这里有15个简单的自我沟通的方法 当我在批评别人指责别人时 我深层地知道对方就是「我」, ...

  • ​LeetCode刷题实战237:删除链表中的节点

    算法的重要性,我就不多说了吧,想去大厂,就必须要经过基础知识和业务逻辑面试+算法面试.所以,为了提高大家的算法能力,这个公众号后续每天带大家做一道算法题,题目就从LeetCode上面选 ! 今天和大家 ...