鱼C论坛

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

[技术交流] Python 数据结构之链表 —— (四)移除重复项及带随机指针的链表复制

[复制链接]
发表于 2020-3-12 13:20:52 | 显示全部楼层 |阅读模式

马上注册,结交更多好友,享用更多功能^_^

您需要 登录 才可以下载或查看,没有账号?立即注册

x
来源于 CSDN


Python 数据结构之链表 —— (四)移除重复项及带随机指针的链表复制


这一篇是 LeetCode 上关于链表的两道题目,难度都是中等,但是我认为难度很大了,尤其是复制链表一题,思路星期。

题目 1

82. 删除排序链表中的重复元素 II

给定一个排序链表,删除所有含有重复数字的节点,只保留原始链表中 没有重复出现 的数字。

示例 1:

输入:1->2->3->3->4->4->5
输出:1->2->5

示例 2:

输入:1->1->1->2->3
输出:2->3


题目解析

要注意 83 题是保留一项,十分简单,就不解析了;而这道题威力加强,如果有两项是重复的,就全都删掉。先看代码再解释:

  1. class Solution:
  2.     def deleteDuplicates(self, head):
  3.         if head is None or head.next is None:
  4.             return head
  5.         fhead = ListNode(0)
  6.         fhead.next = head
  7.         pre = fhead
  8.         cur = head

  9.         while cur:
  10.             while cur.next and cur.val == cur.next.val:
  11.                 cur = cur.next
  12.             if pre.next == cur:  # 如果只此一个值,不删除,pre 后移
  13.                 pre = cur
  14.             else:
  15.                 pre.next = cur.next  # 如果不只一个,删除这些结点,pre 暂不后移
  16.             cur = cur.next
  17.         return fhead.next
复制代码


因为头结点可能会被删掉,先加一个结点;判断相邻两项的值,停留在重复项的最后一个位置(或者就原地不动了),通过 pre 指针与当前指针的关系判断两种情况。

理解解题思想很关键~

题目 2

138. 复制带随机指针的链表
给定一个链表,每个节点包含一个额外增加的随机指针,该指针可以指向链表中的任何节点或空节点。

要求返回这个链表的深拷贝


这一链表结点的定义看下面代码,这一题其实蛮有难度,甚至题目还如此简洁,测试样例也看不懂。

复制问题的关键在于,挨个复制结点时,结点的 next 本身复制了结点间的连接关系并赋予 label 值,创建了一个 label 和 next 属性都一样的新结点。

而 random 指向另一结点时,却不能通过复制 label 来解决,因为其指向一个已存在的结点(该链表中的),你复制的话就分叉了而没有指向该链表中的结点(或者指向了原链表中的结点)。

下面的解决方案有点牛逼,看后面的大神的方法写的:

  1. # Definition for singly-linked list with a random pointer.
  2. # class RandomListNode(object):
  3. #     def __init__(self, x):
  4. #         self.label = x
  5. #         self.next = None
  6. #         self.random = None

  7. class Solution(object):
  8.     def copyRandomList(self, head):
  9.         if head is None:
  10.             return None
  11.         p = head
  12.         while p:
  13.             tmp = p.next
  14.             p.next = RandomListNode(p.label)
  15.             p.next.next = tmp
  16.             p = tmp
  17.         p = head
  18.         while p:
  19.             if p.random:
  20.                 p.next.random = p.random.next
  21.             p = p.next.next
  22.         newhead = RandomListNode(0)
  23.         p = head
  24.         q = newhead
  25.         while p:
  26.             tmp = p.next.next
  27.             q.next = p.next
  28.             q = q.next
  29.             p.next = tmp
  30.             p = tmp
  31.         return newhead.next
复制代码


这一方法用了三次循环,结合下图理解一下。

该方法的时间复杂度为 O(n),空间复杂度 O(1),几乎不占用额外的内存空间。


                               
登录/注册后可看大图


当然这一问题有一个更直观的解决方法,就是哈希表,第一次遍历存储各结点,第二遍设置随机指针,其空间复杂度为 O(n)。

Python 中用字典也可以解决这个问题,可看如下代码:

  1. class Solution:
  2.     # @param head, a RandomListNode
  3.     # @return a RandomListNode
  4.     def copyRandomList(self, head):
  5.         dic = collections.defaultdict(lambda: RandomListNode(0))
  6.         dic[None] = None
  7.         n = head
  8.         while n:
  9.             dic[n].label = n.label
  10.             dic[n].next = dic[n.next]
  11.             dic[n].random = dic[n.random]
  12.             n = n.next
  13.         return dic[head]
复制代码

本帖被以下淘专辑推荐:

小甲鱼最新课程 -> https://ilovefishc.com
回复

使用道具 举报

发表于 2020-3-12 14:51:44 | 显示全部楼层
链表结构是在 Python 里面有定义,还是自定义的?
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

 楼主| 发表于 2020-3-12 14:52:41 | 显示全部楼层
_2_ 发表于 2020-3-12 14:51
链表结构是在 Python 里面有定义,还是自定义的?

自定义的
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2020-3-12 14:57:09 | 显示全部楼层

oh,哪天花点时间讲讲吧
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

发表于 2020-3-12 16:20:25 | 显示全部楼层
本帖最后由 一个账号 于 2020-3-12 18:44 编辑

emm……这个方法可以吗?
  1. n=input()
  2. b=[]
  3. k=0
  4. for i in range(0,len(n)):
  5.                if(n.count(n[i])==1):
  6.                    b.append(n[i])
  7. for i in range(0,len(b)-1):
  8.     print("{0}->".format(b[i]),end='')
  9. print(b[len(b)-1])
复制代码
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

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

本版积分规则

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

GMT+8, 2025-5-14 05:32

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

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