leetcode题目
1.回溯法:
皇后问题,电话键问题,都可以用回溯法解决。
什么是回溯法?
回溯法=“暴力枚举+约束限制条件(这部分是为了从所有结果中只挑出我们需要的结果,可以看成是一种剪枝)。
怎么使用回溯法?
参考我博客的回溯法,用里面的框架来搭代码就行,典型例题有排列树或者子集树。
2.动态规划:
word break问题,
什么是动态规划?
典型动态规划问题=划分状态+递推关系+起始条件。类似于数学归纳法,思路的关键在于找到合适的状态划分。进而找出递推关系,问题自然就迎刃而解。具体可以参考我的博客的动态规划,国王的故事有助于理解动态规划思想。
怎么做动态规划?
找思路的时候:从后往前递推地想。
实现代码的时候:从前往后非递归地写。
3.分治法:
典型例子,二叉树深度的递归求法。
分治法,其实思路跟动态规划基本一样,只不过用递归的方式去解决问题。
总结分治法的框架就是:子状态划分(相当于动态规划的状态划分函数,这个比较关键,找到了,问题基本就解决了,有不少道题都是划分成左半部分和右半部分然后递归)+递归关系式(相当于动态规划的递推关系)+终止条件。
分治法相对难一点。
4.栈:
大多数情况处理字符串,特别是字符串存在特殊字符,比如"/a/./b/../../c/", => "/c"。以及储水池最大容量那种题目。二叉树的前中后序的非递归遍历。
怎么使用栈?
关键点:想清楚怎么将问题用栈的结构来应用到题目+出栈前注意判断栈空。
基本上所有的递归问题都可以用栈来转换,只要适当包装下元素作下标记就行
5.双指针(反向滑动):
典型题目:leetcode的N sum问题。
用双指针作为滑动窗口范围。前后指针异步相向滑动,使得窗口逐渐缩小。
N sum问题都可以用”外层循环+...+最后两层用前后指针“的方法解决。
N sum closest问题可以用”外层循环+...+最后两层用杨氏矩阵”的方法解决。
另外,如果能将N sum转换成two sum,用hash table效果肯定是比较好的。
6.双指针(同向滑动):
典型题目:209题的Binary Search。
用双指针作为滑动窗口范围。前后指针异步同向向前滑动,使得窗口总体一直往前遍历。
7.二分法:
循环数组的查数问题。
第4题求两个有序数组的中位数问题。
一般应用于有序数组、序列。递归和非递归写法都得掌握。如果一个数组是无序的,就要想方法将其变成有序的,然后再进行二分法的应用,比如讲一个无序数组进行前N项和累计构造出一个新的有序数组。比如209题的Binary Search。
8.深度优先搜索DFS:
跟回溯法挺像,代码框架基本都是一样的。要说唯一不同的,就是一般深度优先搜索应用于树、图这种原先各个节点就已经存在特定链接关系的网络中,所以会加上一些是否访问过之类的辅助变量。比如;。
掌握了回溯法,DFS也就基本OK了。
9.广度优先搜索BFS:
主要就是用队列来解决诸如最短距离、最短路径的问题。比如二叉树的层次遍历。
10.图:
掌握邻接矩阵、邻接表的链表和数组形式、并查集。典型题目是任务队列。也就是给出一堆任务和任务依赖关系,求一个任务完成顺序。比如。
11.哈希hash
思想类似函数。STL里面的实现有unordered_set,unordered_multiset, unordered_map,unordered_multimap。
掌握几个常用接口函数:
dict.insert(make_pair(key,value));
dict.erase(key);
dict.count(key);
dict.find(key);
遍历元素:
unordered_set<int>::iterator iter=dict.begin();
set的是*iter;
unordered_map<int,bool>::iterator iter=dict.begin();
map的是iter->first与iter->second;