下面列出最常见的 14 种算法模式,它们可被用于解决常见的问题。另外还会说明如何识别每种模式,并会为每种模式提供一些问题示例。1.滑动窗口 2.二指针或迭代器 3.快速和慢速指针 4.合并区间 5.循环排序 6.原地反转链表 7.树的宽度优先搜索(Tree BFS) 8.树的深度优先搜索(Tree DFS) 9.Two Heaps 10.子集 11.经过修改的二叉搜索 12.前 K 个元素 13.K 路合并 14.拓扑排序
1.滑动窗口 滑动窗口模式是用于在给定数组或链表的特定窗口大小上执行所需的操作,比如寻找包含所有 1 的最长子数组。从第一个元素开始滑动窗口并逐个元素地向右滑,并根据你所求解的问题调整窗口的长度。在某些情况下窗口大小会保持恒定,在其它情况下窗口大小会增大或减小。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 Sliding window --> +------------------------+ | | +----+----+----+----+----+----+----+----+----+ | 1 | 3 | 2 | 6 | -1 | 4 | 1 | 8 | 2 | +----+----+----+----+----+----+----+----+----+ | | +------------------------+ Slide one element forward +------------------------+ | | +----+----+----+----+----+----+----+----+----+ | 1 | 3 | 2 | 6 | -1 | 4 | 1 | 8 | 2 | +----+----+----+----+----+----+----+----+----+ | | +------------------------+
下面是一些你可以用来确定给定问题可能需要滑动窗口的方法:
问题的输入是一种线性数据结构,比如链表、数组或字符串
你被要求查找最长/最短的子字符串、子数组或所需的值
可以使用滑动窗口模式处理的常见问题:
大小为 K 的子数组的最大和(简单)
带有 K 个不同字符的最长子字符串(中等)
寻找字符相同但排序不一样的字符串(困难)
2.二指针或迭代器 二指针(Two Pointers)是这样一种模式:两个指针以一前一后的模式在数据结构中迭代,直到一个或两个指针达到某种特定条件。二指针通常在排序数组或链表中搜索配对时很有用;比如当你必须将一个数组的每个元素与其它元素做比较时。
二指针是很有用的,因为如果只有一个指针,你必须继续在数组中循环回来才能找到答案。这种使用单个迭代器进行来回在时间和空间复杂度上都很低效——这个概念被称为「渐进分析(asymptotic analysis)」。尽管使用 1 个指针进行暴力搜索或简单普通的解决方案也有效果,但这会沿 O(n²) 线得到一些东西。在很多情况中,二指针有助于你寻找有更好空间或运行时间复杂度的解决方案。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 Pointer1 Pointer2 | | V V +----+----+----+----+----+ target sum = 6 | 1 | 2 | 3 | 4 | 6 | +----+----+----+----+----+ 1 + 6 > target sum, therefore let's decrement Pointer2 Pointer1 Pointer2 | | V V +----+----+----+----+----+ | 1 | 2 | 3 | 4 | 6 | +----+----+----+----+----+ 1 + 4 < target sum, therefore let's increment Pointer1 Pointer1 Pointer2 | | V V +----+----+----+----+----+ | 1 | 2 | 3 | 4 | 6 | +----+----+----+----+----+ 2 + 4 == target sum, we have found our pair!
用于识别使用二指针的时机的方法:
可用于你要处理排序数组(或链接列表)并需要查找满足某些约束的一组元素的问题
数组中的元素集是配对、三元组甚至子数组
下面是一些满足二指针模式的问题:
求一个排序数组的平方(简单)
求总和为零的三元组(中等)
比较包含回退(backspace)的字符串(中等)
3.快速和慢速指针 快速和慢速指针方法也被称为 Hare & Tortoise 算法,该算法会使用两个在数组(或序列/链表)中以不同速度移动的指针。该方法在处理循环链表或数组时非常有用。
通过以不同的速度进行移动(比如在一个循环链表中),该算法证明这两个指针注定会相遇。只要这两个指针在同一个循环中,快速指针就会追赶上慢速指针。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 fast,slow | | +----+ +----+ +----+ +----+ +----+ +----+ +--> | 1 | -> | 2 | -> | 3 | -> | 4 | -> | 5 | -> | 6 | --+ +----+ +----+ +----+ +----+ +----+ +----+ | ^ | | | +-----------------------------------+ slow fast | | V V +----+ +----+ +----+ +----+ +----+ +----+ | 1 | -> | 2 | -> | 3 | -> | 4 | -> | 5 | -> | 6 | --+ +----+ +----+ +----+ +----+ +----+ +----+ | ^ | | | +-----------------------------------+ slow fast | | V V +----+ +----+ +----+ +----+ +----+ +----+ | 1 | -> | 2 | -> | 3 | -> | 4 | -> | 5 | -> | 6 | --+ +----+ +----+ +----+ +----+ +----+ +----+ | ^ | | | +-----------------------------------+ fast slow | | V V +----+ +----+ +----+ +----+ +----+ +----+ | 1 | -> | 2 | -> | 3 | -> | 4 | -> | 5 | -> | 6 | --+ +----+ +----+ +----+ +----+ +----+ +----+ | ^ | | | +-----------------------------------+ fast,slow | V +----+ +----+ +----+ +----+ +----+ +----+ | 1 | -> | 2 | -> | 3 | -> | 4 | -> | 5 | -> | 6 | --+ +----+ +----+ +----+ +----+ +----+ +----+ | ^ | | | +-----------------------------------+
如何判别使用快速和慢速模式的时机?
处理链表或数组中的循环的问题
当你需要知道特定元素的位置或链表的总长度时
何时应该优先选择这种方法,而不是上面提到的二指针方法?
有些情况不适合使用二指针方法,比如在不能反向移动的单链接链表中。使用快速和慢速模式的一个案例是当你想要确定一个链表是否为回文(palindrome)时。
下面是一些满足快速和慢速指针模式的问题:
链表循环(简单)
回文链表(中等)
环形数组中的循环(困难)
4.合并区间 合并区间模式是一种处理重叠区间的有效技术。在很多涉及区间的问题中,你既需要找到重叠的区间,也需要在这些区间重叠时合并它们。该模式的工作方式为:
给定两个区间(a 和 b),这两个区间有 6 种不同的互相关联的方式:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 +----------+ +------+ 1) | a | | b | 'a' and 'b' do not overlap +----------+ +------+ +----------+ 2) | a +--+-----+ 'a' and 'b' overlap, 'b' ends after 'a' +-------|--+ | | b | +--------+ +----------+ 3) | a | +-+------+-+ 'a' completely overlap 'b' | b | +------+ +----------+ 4) +-----+--+ a | 'a' and 'b' overlap, 'a' ends after 'b' | +--+-------+ | b | +--------+ +------+ | a | 5) +-+------+-+ 'b' completely overlap 'a' | b | +----------+ +----------+ +------+ 6) | b | | a | 'b' and 'a' do not overlap +----------+ +------+
理解并识别这六种情况有助于你求解范围广泛的问题,从插入区间到优化区间合并等。
那么如何确定何时该使用合并区间模式呢?
如果你被要求得到一个仅含互斥区间的列表
如果你听到了术语「重叠区间(overlapping intervals)
合并区间模式的问题:
5.循环排序 这一模式描述了一种有趣的方法,处理的是涉及包含给定范围内数值的数组的问题。循环排序模式一次会在数组上迭代一个数值,如果所迭代的当前数值不在正确的索引处,就将其与其正确索引处的数值交换。你可以尝试替换其正确索引处的数值,但这会带来 O(n^2) 的复杂度,这不是最优的,因此要用循环排序模式。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 1) Number '2' is not at it's correct place, let's swap it with the correect index. start | V +----+----+----+----+----+----+ | 2 | 6 | 4 | 3 | 1 | 5 | +----+----+----+----+----+----+ start | V +----+----+----+----+----+----+ | 2 | 6 | 4 | 3 | 1 | 5 | +----+----+----+----+----+----+ ^ ^ | | +-----+ 2) After the swap, number '2' is placed at it's correct index. start | V +----+----+----+----+----+----+ | 6 | 2 | 4 | 3 | 1 | 5 | +----+----+----+----+----+----+ 3) Let's move on to the next number. start | V +----+----+----+----+----+----+ | 6 | 2 | 4 | 3 | 1 | 5 | +----+----+----+----+----+----+ 4) Number '2' is at it's correct place, Let's move on the next number. start | V +----+----+----+----+----+----+ | 6 | 2 | 4 | 3 | 1 | 5 | +----+----+----+----+----+----+ ^ ^ | | +-----+ 5) Number '4' is not at it's corrrect place, lets swap it with the next index. start | V +----+----+----+----+----+----+ | 6 | 2 | 3 | 4 | 1 | 5 | +----+----+----+----+----+----+ 6) Number '4' is at it's correct place, Let't move on to the next number. start | V +----+----+----+----+----+----+ | 6 | 2 | 3 | 4 | 1 | 5 | +----+----+----+----+----+----+ ...
如何识别这种模式?
涉及数值在给定范围内的排序数组的问题
如果问题要求你在一个排序/旋转的数组中找到缺失值/重复值/最小值
循环排序模式的问题:
找到缺失值(简单)
找到最小的缺失的正数值(中等)
6.原地反转链表 在很多问题中,你可能会被要求反转一个链表中一组节点之间的链接。通常而言,你需要原地完成这一任务,即使用已有的节点对象且不占用额外的内存。这就是这个模式的用武之地。该模式会从一个指向链表头的变量(current)开始一次反转一个节点,然后一个变量(previous)将指向已经处理过的前一个节点。以锁步的方式,在移动到下一个节点之前将其指向前一个节点,可实现对当前节点的反转。另外,也将更新变量「previous」,使其总是指向已经处理过的前一个节点。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 current previous = null | +-> +----+ +----+ +----+ +----+ +----+ +------+ | 2 | -> | 4 | -> | 6 | -> | 8 | -> | 10 | -> | null | +----+ +----+ +----+ +----+ +----+ +------+ previous current | | +------+ +-> +----+ +-> +----+ +----+ +----+ +----+ +------+ | null | <----- | 2 | | 4 | -> | 6 | -> | 8 | -> | 10 | -> | null | +------+ +----+ +----+ +----+ +----+ +----+ +------+ previous current | | +------+ +----+ +-> +----+ +-> +----+ +----+ +----+ +------+ | null | <- | 2 | <----- | 4 | | 6 | -> | 8 | -> | 10 | -> | null | +------+ +----+ +----+ +----+ +----+ +----+ +------+ ...
如何识别使用该模式的时机:
原地反转链表模式的问题:
反转一个子列表(中等)
反转每个 K 个元素的子列表(中等)
7.树的宽度优先搜索 该模式基于宽度优先搜索(BFS)技术,可遍历一个树并使用一个队列来跟踪一个层级的所有节点,之后再跳转到下一个层级。任何涉及到以逐层级方式遍历树的问题都可以使用这种方法有效解决。
Tree BFS 模式的工作方式是:将根节点推至队列,然后连续迭代知道队列为空。在每次迭代中,我们移除队列头部的节点并「访问」该节点。在移除了队列中的每个节点之后,我们还将其所有子节点插入到队列中。
如何识别 Tree BFS 模式:
如果你被要求以逐层级方式遍历(或按层级顺序遍历)一个树
Tree BFS 模式的问题:
二叉树层级顺序遍历(简单)
之字型遍历(Zigzag Traversal)(中等)
8.树的深度优先搜索 Tree DFS 是基于深度优先搜索(DFS)技术来遍历树。
你可以使用递归(或该迭代方法的技术栈)来在遍历期间保持对所有之前的(父)节点的跟踪。
Tree DFS 模式的工作方式是从树的根部开始,如果这个节点不是一个叶节点,则需要:
决定现在是处理当前的节点(pre-order),或是在处理两个子节点之间(in-order),还是在处理两个子节点之后(post-order)
为当前节点的两个子节点执行两次递归调用以处理它们
如何识别 Tree DFS 模式:
如果你被要求用 in-order、pre-order 或 post-order DFS 来遍历一个树
如果问题需要搜索其中节点更接近叶节点的东西
Tree DFS 模式的问题:
9.TwoHeaps 在很多问题中,我们要将给定的一组元素分为两部分。为了求解这个问题,我们感兴趣的是了解一部分的最小元素以及另一部分的最大元素。这一模式是求解这类问题的一种有效方法。该模式要使用两个堆(heap):一个用于寻找最小元素的 Min Heap 和一个用于寻找最大元素的 Max Heap。该模式的工作方式是:先将前一半的数值存储到 Max Heap,这是由于你要寻找前一半中的最大数值。然后再将另一半存储到 Min Heap,因为你要寻找第二半的最小数值。在任何时候,当前数值列表的中间值都可以根据这两个 heap 的顶部元素计算得到。
识别 Two Heaps 模式的方法:
在优先级队列、调度等场景中有用
如果问题说你需要找到一个集合的最小/最大/中间元素
有时候可用于具有二叉树数据结构的问题
Two Heaps 模式的问题:
10.子集 很多编程面试问题都涉及到处理给定元素集合的排列和组合。子集(Subsets)模式描述了一种用于有效处理所有这些问题的宽度优先搜索(BFS)方法。
该模式看起来是这样:
1 2 3 4 5 给定一个集合 [1, 5, 3] 1.从一个空集开始:[[]] 2.向所有已有子集添加第一个数 (1),从而创造新的子集:[[], [1]] 3.向所有已有子集添加第二个数 (5):[[], [1], [5], [1,5]] 4.向所有已有子集添加第三个数 (3):[[], [1], [5], [1,5], [3], [1,3], [5,3], [1,5,3]]
下面是这种子集模式的一种视觉表示:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 Given set: +----+----+----+ | 1 | 5 | 3 | +----+----+----+ +-----+ | [] | +-----+ copy / \ add 1 V V +-----+-----+ | [] | [1] | +-----+-----+ copy / \ add 5 V V +----+-----+-----+-------+ | [] | [1] | [5] | [1,5] | +----+-----+-----+-------+ copy / \ add 3 V V +----+-----+-----+-------+-----+-------+-------+---------+ | [] | [1] | [5] | [1,5] | [3] | [1,3] | [5,3] | [1,5,3] | +----+-----+-----+-------+-----+-------+-------+---------+
如何识别子集模式:
子集模式的问题:
带有重复项的子集(简单)
通过改变大小写的字符串排列(中等)
11.经过修改的二叉搜索 只要给定了排序数组、链表或矩阵,并要求寻找一个特定元素,你可以使用的最佳算法就是二叉搜索。这一模式描述了一种用于处理所有涉及二叉搜索的问题的有效方法。
对于一个升序的集合,该模式看起来是这样的:
1 2 3 4 1.首先,找到起点和终点的中间位置。寻找中间位置的一种简单方法是:middle = (start + end) / 2。但这很有可能造成整数溢出,所以推荐你这样表示中间位置:middle = start + (end — start) / 2。 2.如果键值(key)等于中间索引处的值,那么返回这个中间位置。 3.如果键值不等于中间索引处的值: 4.检查 key < arr[middle] 是否成立。如果成立,将搜索约简到 end = middle — 15.检查 key > arr[middle] 是否成立。如果成立,将搜索约简到 end = middle + 1
下面给出了这种经过修改的二叉搜索模式的视觉表示:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 Search 'key' = '5' start middle end | | | V V V +----+----+----+----+----+----+----+ | 1 | 2 | 3 | 4 | 5 | 6 | 7 | +----+----+----+----+----+----+----+ As key > arr[middle], therefore start = middle + 1 start middle end | | | V V V +----+----+----+----+----+----+----+ | 1 | 2 | 3 | 4 | 5 | 6 | 7 | +----+----+----+----+----+----+----+ As key < arr[middle], therefore end = middle - 1 start,middle,end | V +----+----+----+----+----+----+----+ | 1 | 2 | 3 | 4 | 5 | 6 | 7 | +----+----+----+----+----+----+----+ As key == arr[middle], return middle as the required index
经过修改的二叉搜索模式的问题:
与顺序无关的二叉搜索(简单)
在经过排序的无限数组中搜索(中等)
12.前K个元素 任何要求我们找到一个给定集合中前面的/最小的/最常出现的 K 的元素的问题都在这一模式的范围内。
跟踪 K 个元素的最佳的数据结构是 Heap。这一模式会使用 Heap 来求解多个一次性处理一个给定元素集中 K 个元素的问题。该模式是这样工作的:
1.根据问题的不同,将 K 个元素插入到 min-heap 或 max-heap 中 2.迭代处理剩余的数,如果你找到一个比 heap 中数更大的数,那么就移除那个数并插入这个更大的数
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 (1) +----+----+----+----+----+----+ +----+ | 3 | 1 | 5 | 12 | 2 | 11 | | 1 | +----+----+----+----+----+----+ +----+ / \ V V Insert the first three +----+ +----+ numbers in the heap | 5 | | 3 | +----+ +----+ (2) +----+----+----+----+----+----+ +----+ | 3 | 1 | 5 | 12 | 2 | 11 | | 3 | +----+----+----+----+----+----+ +----+ / \ V V The root is smaller than '12' +----+ +----+ so take '1' out and inset '12' | 5 | | 12 | +----+ +----+ (3) +----+----+----+----+----+----+ +----+ | 3 | 1 | 5 | 12 | 2 | 11 | | 3 | +----+----+----+----+----+----+ +----+ / \ V V Skip '2', as it is not bigger +----+ +----+ than the root '3' | 5 | | 12 | +----+ +----+ (4) +----+----+----+----+----+----+ +----+ | 3 | 1 | 5 | 12 | 2 | 11 | | 5 | +----+----+----+----+----+----+ +----+ / \ V V The root is smaller than '12' +----+ +----+ so take '5' out and inset '12' | 11 | | 12 | +----+ +----+
这里无需排序算法,因为 heap 将为你跟踪这些元素。
如何识别前 K 个元素模式:
如果你被要求寻找一个给定集合中前面的/最小的/最常出现的 K 的元素
如果你被要求对一个数值进行排序以找到一个确定元素
前 K 个元素模式的问题:
前面的 K 个数(简单)
最常出现的 K 个数(中等)
13.K路合并 K 路合并能帮助你求解涉及一组经过排序的数组的问题。
当你被给出了 K 个经过排序的数组时,你可以使用 Heap 来有效地执行所有数组的所有元素的排序遍历。你可以将每个数组的最小元素推送至 Min Heap 以获得整体最小值。在获得了整体最小值后,将来自同一个数组的下一个元素推送至 heap。然后,重复这一过程以得到所有元素的排序遍历结果。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 (1) +----+----+----+ +----+ L1 |-2--| 6 | 8 | | 1 | +----+----+----+ +----+ +----+----+----+ / \ L2 |-3--| 6 | 7 | V V +----+----+----+ +----+ +----+ +----+----+----+ | 2 | | 3 | L3 |-1--| 3 | 4 | +----+ +----+ +----+----+----+ Inset the first number from each array in the teap (2) +----+----+----+ +----+ L1 |-2--| 6 | 8 | | 2 | +----+----+----+ +----+ +----+----+----+ / \ L2 |-3--| 6 | 7 | V V +----+----+----+ +----+ +----+ +----+----+----+ | 3 | | 3 | L3 |-1--|-3--| 4 | +----+ +----+ +----+----+----+ +----+ Merged List: | 1 | +----+ (3) +----+----+----+ +----+ L1 |-2--|-6--| 8 | | 3 | +----+----+----+ +----+ +----+----+----+ / \ L2 |-3--| 6 | 7 | V V +----+----+----+ +----+ +----+ +----+----+----+ | 6 | | 3 | L3 |-1--|-3--| 4 | +----+ +----+ +----+----+----+ +----+----+ Merged List: | 1 | 2 | +----+----+
该模式看起来像这样:
1 2 3 4 1.将每个数组的第一个元素插入 Min Heap 2.之后,从该 Heap 取出最小(顶部的)元素,将其加入到合并的列表。 3.在从 Heap 移除了最小的元素之后,将同一列表的下一个元素插入该 Heap 4.重复步骤 2 和 3,以排序的顺序填充合并的列表
如何识别 K 路合并模式:
具有排序数组、列表或矩阵的问题
如果问题要求你合并排序的列表,找到一个排序列表中的最小元素
K 路合并模式的问题:
合并 K 个排序的列表(中等)
找到和最大的 K 个配对(困难)
14.拓扑排序 拓扑排序可用于寻找互相依赖的元素的线性顺序。比如,如果事件 B 依赖于事件 A,那么 A 在拓扑排序时位于 B 之前。
这个模式定义了一种简单方法来理解执行一组元素的拓扑排序的技术。
该模式看起来是这样的:
1 2 3 4 5 6 7 8 9 10 1.初始化。a)使用 HashMap 将图(graph)存储到邻接的列表中;b)为了查找所有源,使用 HashMap 记录 in-degree 的数量 2.构建图并找到所有顶点的 in-degree。a)根据输入构建图并填充 in-degree HashMap 3.寻找所有的源。a)所有 in-degree 为 0 的顶点都是源,并会被存入一个队列 4.排序。 a)对于每个源,执行以下操作: i)将其加入到排序的列表; ii)根据图获取其所有子节点; iii)将每个子节点的 in-degree 减少 1; iv)如果一个子节点的 in-degree 变为 0,将其加入到源队列。 b)重复 (a),直到源队列为空。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 +----+ +----+ | 1 | | 1 | +----+ +----+ / \ / | V V V | +----+ +----+ | | 2 |-\ | 3 | | +----+ \ +----+ | / \ \/ | / \ /\ | V V V \ V +----+ +----+ \--> +----+ | 2 | | 3 | | 3 | +----+ +----+ +----+ Add all sources to the sorted list. Remove all sources and their edges to find new souorces +----+ +----+ | 2 | | 2 | +----+ +----+ / | \ | V V V V Sources: [3,4] +----+ +----+ +----+ Topological Sort: "5,6" | 2 | | 3 | | 3 | +----+ +----+ +----+ Add all sources to the sorted list. Remove all sources and their edges to find new souorces +----+ +----+ +----+ | 2 | | 3 | | 3 | +----+ +----+ +----+ Sources: [0,1,2] Topological Sort: "5,6,3,4" All remaining vertices are souce, so we will and them in the sourted list Sources: [] Topological Sort: "5,6,3,4,0,1,2"
如何识别拓扑排序模式:
处理无向有环图的问题
如果你被要求以排序顺序更新所有对象
如果你有一类遵循特定顺序的对象
拓扑排序模式的问题:
参考:Facebook 工程师总结的 14 种算法面试模式