Leetcode 【287、1035】

题目描述:【Two Pointers】287. Find the Duplicate Number
解题思路:

这道题由于 Note 中有很多限制,想到的一些方法(如排序后再找、开辟 O(n) 空间计数、双循环)都不满足题意。由于只有一个数字出现了 2 次或 2 次以上,因此想到可不可以像 Leetcode 142. Linked List Cycle II 那样,使用快慢指针的方法去解决(因为只有一个重复数字出现,因此可以理解成有环)?

答案为 Yes。

我们已经知道,判断一个链表是否有环可以像 Leetcode 【Two Pointers】141. Linked List Cycle 那样,使用快慢指针。慢指针一次走一步,快指针一次走两步,两指针相遇就证明有环。

但是在 142 中,我们要确定环的入口在哪里,怎么做呢?还是先像 141 那样,找到第一次快慢指针相遇的地方。然后,将慢指针重新指向头结点,快指针依然指向相遇的那个节点。两个指针以每次一步的速度来遍历,直到他们再次相遇。此时,他们相遇的节点即是链表环的入口(这个问题的证明留到我做 142 题时再写)。

那么回到 287 这道题。我们只需要构造出类似于 142 题的“链表”,然后使用上述方法就可以找到重复的那个数字(即环的入口)。

题目大致意思是有 n+1 空间的整形数组,里面存的是 1∼n,而且这个数组里面有且仅存在一种重复的数字(重复但不限于重复一次),这里因为题目的特殊性,我们可以拿数组的索引号和数组里面存放的数字做文章。 因为数组中的数字是不大于 n 的,所以也就意味着不大于索引号(0∼n),所以在每次读取一个数组中的数字内容时,我们可以将这个数字作为新的索引,相当于现在我们可以构造出一个有向循环图,包含n+1 个节点和 n+1 条边,所以必定有环。

举个例子,nums = [3,1,3,4,2],下标从 0 开始。首先第一个数字 3 指向下标 3 的数字 4,数字 4 指向下标 4 的数字 2,数字 2 指向下标 2 的数字 3,数字 1 指向下标 1 的数字 1。那么就会得到如下的有向循环图:

从这里看出来 3,4,2形成了一个环,而环的入口点是 3,这个问题就转换成了 Leetcode 142 的问题了。

当然这里还有一个问题,看上图中,1 是单独出来的,那么当我们初始化的时候,如果刚好碰到像 1 这种单独成环的怎么办?其实我们从数组第 1 个点进去即可,即下标为 0 的数字。因为题目说了数组中数字的范围是 1-n 的,所以 0 点处的值是不可能单独成环的,放心的从第一个数字开始找环即可。

Python3 实现:
class Solution:
    def findDuplicate(self, nums: List[int]) -> int:
        slow = nums[nums[0]]  # 慢指针走一步
        fast = nums[nums[nums[0]]]  # 快指针走两步
        while slow != fast:  # 找到第一次相遇的地方
            slow = nums[slow]
            fast = nums[nums[fast]]
        slow = nums[0]  # 慢指针指向链表入口
        while slow != fast:  # 快慢指针走都走一步,就能在环的入口处相遇
            slow = nums[slow]
            fast = nums[fast]
        return slow

print(Solution().findDuplicate([2,3,1,5,3,6,4])  # 3 (画的链表为 2->1->3->5->6->4>3)

题目描述:【DP】1035. Uncrossed Lines
解题思路:

刚开始想了一下,发现应该用动态规划求解。在构造矩阵、填写最优子结构的结果的过程中,发现怎么和最长公共子序列的转移方程一模一样。再去理解一下题目,好吧,就是求最长公共子序列。代码 AC,结束。

最长公共子序列的参考博客:常考的经典算法--最长公共子序列(LCS)与最长公共子串(DP)

Python3 实现:
class Solution:
    def maxUncrossedLines(self, A: List[int], B: List[int]) -> int:
        dp = [[0] * (len(B)+1) for _ in range(len(A)+1)]
        for i in range(1, len(A)+1):
            for j in range(1, len(B)+1):
                if A[i-1] == B[j-1]:
                    dp[i][j] = dp[i-1][j-1] + 1
                else:
                    dp[i][j] = max(dp[i-1][j], dp[i][j-1])
        return dp[-1][-1]
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,496评论 6 501
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,407评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,632评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,180评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,198评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,165评论 1 299
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,052评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,910评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,324评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,542评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,711评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,424评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,017评论 3 326
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,668评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,823评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,722评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,611评论 2 353

推荐阅读更多精彩内容