基本算法
1、枚举
从问题的所有可能解的集合中一一枚举,判断能使命题成立的解。
2、模拟
根据题意一步步走代码,通常会设坑。
POJ1298
POJ2632
POJ1922
POJ1068
3、二分
二分查找
在一个单调有序集合中查找元素,每次将集合分为左右两部分,判断解在哪个部分并调整上下界,直到找到目标。
k = (low+top)/2
array[k] == key -------succeed
array[k] > key ---------array[low ~ k-1]
array[k] < key ---------array[k+1 ~ top]
二分答案+检验
可以确定解在一个有序的范围内,用二分查找的方式查找正确的解。
每次判断都采用一个检验函数。
ZOJ4130
POJ2785
HihoCoder1139
HDU5038
POJ1018
4、并查集
初始化每个节点的父节点是自己
查询2个点是否在同一集合:比较最高父节点是否相同
合并:一个节点的最高父节点等于另一个节点
优化:使每个点的直接父节点是最高父节点
5、DFS
一条路走到黑,并标记走过的点,走不了就回退。
POJ2386
百练2815
百练4103
百练4123
POJ3083
POJ1321
6、BFS
7、DP
将一个问题分解成许多子问题。
将问题转化为01背包
完全背包改一下初始化
//不压缩
for(int i=1;i<=m;i++)
for(int j=n;j>=w[i];j--)
dp[i][j] = max(dp[i-1][j] , dp[i-1][j-w[i]] + v[i]);
//滚动数组
for(int i=1;i<=m;i++)
{
int tap=i%2;
for(int j=n;j>=w[i];j--)
dp[tap][j] = max(dp[!tap][j] , dp[!tap][j-w[i]] + v[i])
}
//一维数组
for(int i=1;i<=m;i++)
for(int j=n;j>=w[i];j--)
dp[j] = max(dp[j] , dp[j-w[i]] + v[i])