You have solved 489/1771 problems. 主要追求两个目标:代码简短,时间空间复杂度低。
注记
4-sum
使用了multimap
,时间复杂度
合法的四元组有三类:
a<=b<c<=d
,枚举c
和d
,判断是否存在和为target-c-d
的二元组a<=b=c<=d
,枚举b
和d
,判断是否存在target-b-b-d
a=b=c<=d
,枚举a
,判断是否存在target-a-a-a
分别统计,小心实现可以保证不会产生相同的候选解,从而无需去重。
Alien Dictionary
对于每一对单词,可以确定去除最长公共前缀后的下一对字母的大小关系。两个单词的最长公共前缀等于夹在其中的(所有相邻单词对的最长公共前缀)的最小值。根据相邻单词对的信息即可推得所有任意单词对的信息。因此只需根据相邻单词对求出拓扑关系。
Basic Calculator
这是operator-precedence grammar,可以用operator-precedence parser计算。这类算法的一个特例是shunting-yard algorithm,适用于本题。 为了方便,我假设字串开头和末尾都有隐含的'\0'字符。使用的文法如下:
1 | S := %x00 E %x00 |
参考http://www.scribd.com/doc/51358638/16/Operator-Precedence-Relations。
在比较两个操作符时,区分左手边和右手边,左手边用于先处理的操作符,右手边为新来的操作符。使用左手边操作符的in-stack precedence(isp)和右手边操作符的incoming precedence(icp)进行比较。立即数可以看作icp很大,因此shift后可以立刻reduce。
Best Time to Buy and Sell Stock IV
线性解法。见2015-03-27-leetcode-best-time-to-buy-and-sell-stock-iv。
Closest Binary Search Tree Value II
题意:求BST中与目标值最接近的k个节点。
BST中关键字小于等于目标值的所有节点可以用若干(不带右孩子的子树)表示。这些(不带右孩子的子树)在一条链上,可以用栈维护,得到用于计算前驱的的迭代器。求出目标值的k个前驱的时间复杂度为
Contains Duplicate III
以
Course Schedule
拓扑排序。入度减为0的顶点即成为候选,可以把入度向量组织成单链表,从而无需使用额外的队列或栈存储候选顶点,减小空间占用。
Dungeon Game
动态规划,对于每个格子记录从它出发顺利到达终点需要的最小血量。从
注意对于每个格子其实有进和出两个状态:进入该格子时(未考虑当前格子影响);和离开该格子后(已考虑当前格子影响)。这两种方法都行,考虑到程序的结构可能有三部分:初始化边界条件、状态转移、根据状态表示答案,需要仔细斟酌以追求代码简短。这里两种方式并无明显代码长度差异。
Expression Add Operators
见2015-10-16-leetcode-expression-add-operators。
Find Minimum in Rotated Sorted Array
方法是二分,如果要写短可以采用下面的思路:
子数组中如果有相邻两个元素a[i]>a[i+1]
,则a[i+1]
是整个数组的最小值。
若子数组a[i..j]
满足a[i]>a[j]
,则存在i<=k<j
使得a[k]>a[k+1]
。
对于子数组a[l..h]
判断a[l]>a[m]
或a[m]>a[h]
是否有一个成立,成立则可以去除一半候选元素,不然a[h]>a[l]
为一个逆序对。
Find Minimum in Rotated Sorted Array II
留意几个特殊情形:[1,0,1,1,1]
、[1,1,1,0,1]
。
对于子数组a[l..h]
,令m=floor(l+h)/2
(注意m
可能等于l
)。判断a[l]>a[m]
或a[m]>a[h]
是否有一个成立,成立则可以去除一半候选元素。
若两个条件均不满足则a[l]<=a[m]<=a[h]
,若a[h]>a[l]
则说明a[l]
最小,否则a[l]=a[m]=a[h]
,可以把范围缩小为a[l+1..h-1]
。
另外一种方法是只判断a[m]
与a[h]`的大小关系。
Find Peak Element
二分查找,把区间缩小为仍包含候选值的子区间。对于相邻两个元素1
2
3
4
5
6
7int l = 0, h = a.size();
while (l < h-1) {
int m = l+h >> 1;
if (a[m-1] > a[m]) h = m;
else l = m;
}
return l;
或者检查中间元素是否满足要求,可以提前退出循环。 1
2
3
4
5
6
7
8int l = 0, h = a.size();
while (l < h-1) {
int m = l+h >> 1;
if (a[m-1] > a[m]) h = m;
else if (m+1 == h || a[m] > a[m+1]) l = h = m;
else l = m+1;
}
return l;
First Missing Possitive
空间复杂度
Fraction to Recurring Decimal
注意INT_MIN/(-1)
会溢出。
Graph Valid Tree
- 有
条边且没有simple circle。可以用union-find algorithm判断。 - 联通且无simple circle。可以用graph traversal。
Guess Number Higher or Lower II
动态规划优化,
Invert Binary Tree
选择一种binary tree traversal算法,节点访问改成交换左右孩子即可。 如果要O(1)空间的话,得借鉴Morris pre/in-order traversal的思路,该算法的核心是通过左子树最右节点的右孩子是否指向当前节点来判断当前节点被访问到的次数,从而决定当前应该进行的操作(探索左子树或访问当前节点)。但交换子树操作会弄乱左子树最右节点的位置,因此pre-order和in-order都行不通,但post-order可行。Morris post-order traversal(这个叫法也许不恰当)可以参考https://www.quora.com/What-is-a-good-way-to-implement-stackless-recursion-less-post-order-traversal-for-a-non-threaded-binary-tree-using-Morris-method的描述,pre/in-order的访问是针对一个节点进行的,但post-order则需要对当前节点左子树的右链进行,并且需要翻转两次以达到按post-order输出的效果。但这里只需要保证每个节点都被访问一次,不需要严格的post-order顺序,因此无需翻转操作。
Jump Game
空间复杂度
Jump Game II
使用output-restricted
queue优化的动态规划,注意到动态规划值的单增性可以用类似BFS的方式,空间复杂度
Lexicographical Numbers
在trie中找1~n,存在迭代解法。
Linked List Cycle II
单链表找圈。Brent's cycle detection algorithm或Floyd's cycle detection algorithm。
如果只需要判断是否有圈,而不用指出圈的起点,可以使用pointer reversal。访问链表的过程中翻转链表,倘若访问到NULL,则说明无圈;若回到了起点,则说明有圈。如果有圈的话,算法执行完毕后,圈的方向会被反转,可以再执行一次翻转过程还原。下面是这个方法的实现:
1 | // return true if a cycle is detected |
Longest Palindromic Substring
线性时间求出以每个位置为中心的最长回文子串的Manacher's algorithm。http://www.csie.ntnu.edu.tw/~u91029/Palindrome.html提供的描述感觉最清晰。但实现我更喜欢我从某本stringology书上看到的。
Majority Element
Boyer-Moore majority vote algorithm http://www.cs.utexas.edu/~moore/best-ideas/mjrty/,思路是每次找到两个不同的元素并丢弃,这样做不会丢失解。容器里只剩下一种元素时,它就是majority。
Majority Element II
Boyer-Moore majority vote algorithm的扩展,找出所有出现次数大于
Maximum Gap
平均数原理,求出极差
Maximum Subarray
Kadane's algorithm
Meeting Rooms II
题意中两条线段冲突当且仅当它们的公共部分长度大于0。 这是interval graph,所求的是chromatic number,interval graph包含与perfect graph,因此chromatic number等于maximum clique顶点数,即最大线段相交数,可以把所有端点排序计算。 或者按起始端点排序后用贪心算法:维护若干线段集合,按起始端点总从小到大考虑每个线段,任选一个不冲突的集合加入,若无不冲突的集合则自成一个新集合。最后,集合数即为答案。实现时,每个集合用最大右端点值表示,所有集合组织为一个小根binary heap,若当前考虑线段的起始点小于根则压入一个新元素,否则修改根的值。
Merge k Sorted Lists
Tournament sort,可以用priority_queue
实现。
Min Stack
使用两个栈,原栈S
存放各个元素。每当新元素小于等于当前最小值时,就把新元素复制一份压入另一个栈S'
。弹出时,若元素在S'
中出现,则也从S'
中弹出。
另一种方法是记录新元素与当前最小值的差值,每个元素需要多记录1个bit。可惜C++会Memory Limit Exceeded,感觉不合理。
Minimum Window Substring
尺取法
Missing Number
从左到右考虑每个元素,若不等于它的位置则把它交换到正确的位置并重复此过程。
N Queen
bitmask存储列,正反斜线控制的格子。
Number of 1 Bits
求popcount。__builtin_popcount()
或者网上查找各种bit
twiddling hacks。
One Edit Distance
编辑距离为1有三种可能:S比T多一个字符、T比S多一个字符、S与T长度相同且有一个位置不同。除去最长公共前缀和最长公共后缀,三种可能的检查方式可以统一为判断较长的串长度是否为1。
Palindrome Partitioning
f[i][j] = calc(f[ii][jj] : i <= ii <= jj <= j)
形式的动态规划可以采用如下计算方式。
1 | for (int i = n; --i >= 0; ) |
Palindrome Partitioning II
求原串的最少回文串划分。O(n^2)时间,可以优化到O(n)空间。
我没能找到比O(n^2)快的算法。相关的问题有检测一个串是否分割成k个回文串。k小于等于4时均有线性算法,参见Text Algorithms一书Theorem 8.17。k大于4时不确定。
Perfect Rectangle
使用unordered_map
的
或者判断所有小矩形面积和等于四个极点围成的矩形的面积,并且小矩形没有重叠(用sweep line判断)。
Perfect Squares
https://leetcode.com/discuss/57477/sqrt-applying-fermats-theorm-brahmagupta-fibonacci-identity
由Lagrange's four-square theorem知答案为1~4,由Legendre's
three-square theorem检查答案是否为4。若小于4,则再
另有期望
Reconstruct Itinerary
题目即lexicographically smallest Eulerian path。Eulerian path最简单的算法是Hierholzer's algorithm,有向图中仅有一个顶点出度=入度+1,记为源;一个顶点入度=出度+1,记为汇。任求一条源到汇的路径,之后把各个圈合并到路径上即可。本题要求字典序最小,对Hierholzer's algorithm稍加变形即可。
从源开始,贪心地选择序号最小的下一个顶点,删除这条边,重复此过程,即得到一条源到汇的路径。需要把圈添加到该路径上。若一个圈只包含路径上的一个顶点,那么在该顶点处添加圈即可。若一个圈包含路径上的多个顶点,那么应该把它放在离汇最近的顶点上,这样能保证字典序最小。
Recover Binary Search Tree
使用Morris in-order traversal找到邻接失序对。
如果交换的元素相邻,则有一个邻接失序对(如0 1 2 3 4 ->
0
1 3 2 4),否则有两个邻接失序对(如
0 1 2 3 4 5 ->
0 4 2 3 1 5
)。
Remove Nth Node From End of List
http://meta.slashdot.org/story/12/10/11/0030249/linus-torvalds-answers-your-questions
使用pointers-to-pointers很多时候能简化实现。
Regular Expression Matching
P
为模式串,T
为文本,
f[i][j]
表示P
的前i
个字符能否匹配T
的前j
个字符。
根据f[i-1][*]
计算f[i][*]
。
这个方法也可以看作构建了精简表示的Thompson's automaton,时间复杂度
Remove Element
原地修改数组,我喜欢把这种操作称作deflate。
Remove Invalid Parentheses
最少移除个数很容易计算,把可以配对的左右括号全部抵消,剩余括号数即是(形如:)))...(((
)。难点在于输出所有不同答案,多数解法使用DFS枚举要删除哪些括号,枚举量大于答案个数。
有一个枚举量固定为答案个数的方法,找到残余串)))...(((
中右括号与左括号的分割点。DFS分割点左半部分,保证到达残余串第
Repeated DNA Sequences
可以用类似Rabin-Karp的rolling hash方法,求出所有长为10的子串的hash值,判断是否有重复。只有四个字符,长度为10,故可以使用polynomial hash code。 另外也可以用suffix tree/suffix array等方法,找longest common prefix大于等于10的两个后缀。注意不能重复输出字串。
Reverse Bits
Hacker's Delight (2nd) 7.1 Reversing Bits and Bytes。
Rotate Array
经典的三次reverse,或者拆成gcd(k,n)
个置换群循环右移一格。
Same Tree
类似于same fringe problem,可以试验generator、co-routine、lazy evaluation、continuation等。如果要O(1)的空间复杂度可以用Morris in-order traversal等方法。
Set Matrix Zeroes
Search For a Range
设序列用区间表示为[begin,end)
。使用C++
STL风格的lower_bound、
upper_bound。有几点好处:
- 循环内只有一次比较
- 循环退出后两个指针相等,且落在
[begin,end]
,都是迭代器的合法取值。某些binary search写法返回值可能为begin-1
,某些容器不合法 [lower_bound(a[],x), upper_bound(a[],x)
给出了元素x
所在的子区间
Search Insert Position
C++ <algorithm>
的lower_bound
。
Serialize and Deserialize Binary Tree
把NULL视作叶子,则pre/in/post-order
traversal可以用于编码。若使用Morris traversal则额外空间为
Shortest Palindrome
找出原串的最长回文前缀,用Manacher's algorithm求解,把剩下部分复制一份、翻转后、接到左边,就得到了最短的回文串。求最长回文前缀的另一种方法是把原串以及翻转用特殊字符隔开,用Morris-Pratt algorithm求border,该值即为最长回文前缀长度。
Single Number III
所有数的xor等于两个出现一次的数的xor,不妨设为k
,则k & -k
为二进制表示上某个为1的数位。根据这个数位把所有元素划分成两份,每份只有一个出现一次的数。
Smallest Good Base
设
Sort Colors
Dutch national flag problem。如果不要求000111222,允许111000222111,那么有交换次数更少的Bentley-McIlroy算法http://www.iis.sinica.edu.tw/~scm/ncs/2010/10/dutch-national-flag-problem-3/
Sqrt(x)
Hacker's Delight (2nd) 11.1.1。
牛顿迭代法。46340 = floor(sqrt(INT_MAX))
Scramble String
我用了
Sort List
我用了较麻烦的natural merge sort。Quicksort实现会简单很多。
Strobogrammatic Number III
定义一个函数,用于计算小于给定数的strobogrammatic number个数,则用减法即可得到题目所求。下面考虑如何计算这个函数。 小于给定数high的strobogrammatic number分为两类:
- 位数不够的,可看作有若干前导0。容易求得。
- 位数相同。可以枚举与high的公共前缀的长度,再计算下一个数位小于high中对应数位的strobogrammatic
number个数。由于位数为
的strobogrammatic number只有 个数位是独立的,枚举的公共前缀长度得小于 。
Sudoku Solver
转化为exact cover problem,使用dancing links + Algorithm X求解。
Trapping Rain Water
两个指针夹逼,空间
Unique Binary Search Trees II
类似动态规划的思想,子树可以复用。
Wiggle Sort
从左到右考虑每一对相邻元素,如果不满足大小关系则交换,交换不会影响之前一对元素的大小关系。
Word Ladder
基本方法是BFS或bidirectional BFS,相邻关系的确定有两种思路:枚举所有其他顶点判断是否相邻、枚举顶点的变异判断是否为顶点。对于第一种思路,还可以利用一个技巧:把所有字串都分为左右两半,若两个字串hamming distance为1,则左半部分相同或右半部分相同。
Verify Preorder Sequence in Binary Search Tree
对于前序序列中的递减连续段,它们形成了BST的某个左孩子链。
Maximal Rectangle
秋叶拓哉(iwi)、岩田阳一(wata)和北川宜稔(kita_masa)所著,巫泽俊(watashi)、庄俊元(navi)和李津羽(itsuhane)翻译的《挑战程序设计竞赛》
逐行扫描棋盘,对于每一行
表示第 列上方连续为1的行数: h[j] = a[i][j] == 0 ? 0 : h[j]+1;
- 若
,令 为 ;若 则设为 - 若
,令 为 ;若 则设为
1
2
3
4
5
6
7if a[i][j] == 0:
l[j] = j
elif h[j] == 1:
l[j] = j-(a[i][j]及左边连续的1的个数)+1
else
# 下式中右边的l[j]是第i-1行计算得到的l[j]
l[j] = min(l[j], j-(a[i][j]及左边连续的1的个数)+1)
1 | #define ROF(i, a, b) for (int i = (b); --i >= (a); ) |
潘宇超 2008
1 | #define ROF(i, a, b) for (int i = (b); --i >= (a); ) |
ACRush某TopCoder SRM
逐行扫描棋盘,对于每一行记录
对于每一行1
2
3
4x = [0,0,...,0]
foreach j, h[j]: # 从大到小遍历h[j]
x[j] = 1
ans = max(ans, (max(k : x[j..k]都为1) - min(k : x[k..j]都为1)) * h[j])
上面伪代码可以在
不相交集合可以用union-find算法,但针对这种特殊情形存在
1 | #define ROF(i, a, b) for (int i = (b); --i >= (a); ) |
栈维护histogram
逐行扫描棋盘,对于每一行记录
对于每一行
其他方法
枚举两列,求出夹在两列间的最大1子矩形。
令
把0看成障碍点,求不包含障碍点的最大子矩形。
把障碍点按横座标排序。枚举障碍点作为候选子矩形的左端,向右扫描各个障碍点,维护纵座标和枚举点最近的上下个一个障碍点。枚举点、扫描线和上下障碍点确定了候选子矩形的边界。
若0的数目为,
代码索引
1 | copy($('td:has(.ac) ~ td:nth-child(3) a').map(function(_,e){ |
1 | require 'mechanize' |