一般ACM或者笔试题的时间限制是1秒或2秒。
在这种情况下,C++代码中的操作次数控制在 1 0 7 ∼ 1 0 8 10^7∼10^8 107∼108 为最佳。
下面给出在不同数据范围下,代码的时间复杂度和算法该如何选择:
指数级别, dfs+剪枝,状态压缩dp
floyd,dp,高斯消元
dp,二分,朴素版Dijkstra、朴素版Prim、Bellman-Ford
块状链表、分块、莫队
各种sort,线段树、树状数组、set/map、heap、拓扑排序、dijkstra+heap、prim+heap、Kruskal、spfa、求凸包、求半平面交、二分、CDQ分治、整体二分、后缀数组、树链剖分、动态树
单调队列、 hash、双指针扫描、并查集,kmp、AC自动机
,常数比较小的 O ( n l o g n ) O(nlogn) O(nlogn) 的做法:sort、树状数组、heap、dijkstra、spfa
双指针扫描、kmp、AC自动机、线性筛素数
判断质数
最大公约数,快速幂,数位DP
高精度加减乘除
k表示位数,高精度加减、FFT/NTT
int数据范围
:-2147483648~2147483647、大约 2 ∗ 1 0 9 2*10^9 2∗109左右long long数据范围
:正负 1 0 18 10^{18} 1018之间cin、cout
scanf、printf
队列,bfs。栈,dfs,递归。
《深入理解计算机系统》
bfs: 数据结构:queue、空间:O(2^h)空间大、可以找到一条合法路径,最小步数,一层一层进行拓展,最短路。
dfs: 数据结构:stack、空间:O(n)空间小、不能找最短路径,不具有最短路性质
思路奇怪的:dfs
最短距离:bfs
技巧:
4方向偏移量:
int dx[4] = {-1, 0, 1, 0}, dy[4] = {0, 1, 0, -1};
8方向偏移量:
int dx[8] = {-1, -1, -1, 0, 1, 1, 1, 0};
int dy[8] = {-1, 0, 1, 1, 1, 0, -1, -1};
8方向日字形偏移量:
int dx[8] = {-2, -1, 1, 2, 2, 1, -1, -2};
int dy[8] = {1, 2, 2, 1, -1, -2, -2, -1};
注意: 对于多组输入的题目,不要忘记将判重数组初始化memset
深搜注意爆栈
深搜恢复现场:一个状态需要多次使用时才会恢复现场。
回溯、剪枝
关键:考虑顺序,遍历全部方案。
回溯时恢复现场st[]数组
path[]存路径
Flood Fill、图与树的遍历、
深搜的返回条件
//迷宫问题:是否走到最后一个点
if(x==xb&&y==yb) return true;
//其他类搜索问题:是否搜索到最后一个点
if(cnt==n*m) {ans++;return;}
//搜索到没有解
cnt+=dfs(a,b);
假设有 3 个空位,从前往后填数字,每次填一个位置,填的数字不能和前面一样。
最开始的时候,三个空位都是空的:__ __ __
首先填写第一个空位,第一个空位可以填 1,填写后为:1 __ __
填好第一个空位,填第二个空位,第二个空位可以填 2,填写后为:1 2 __
填好第二个空位,填第三个空位,第三个空位可以填 3,填写后为: 1 2 3
这时候,空位填完,无法继续填数,所以这是一种方案,输出。
然后往后退一步,退到了状态:1 2 __ 。剩余第三个空位没有填数。第三个空位上除了填过的 3 ,没有其他数字可以填。
因此再往后退一步,退到了状态:1 __ __。第二个空位上除了填过的 2,还可以填 3。第二个空位上填写 3,填写后为:1 3 __
填好第二个空位,填第三个空位,第三个空位可以填 2,填写后为: 1 3 2
这时候,空位填完,无法继续填数,所以这是一种方案,输出。
然后往后退一步,退到了状态:1 3 __ 。剩余第三个空位没有填数。第三个空位上除了填过的 2,没有其他数字可以填。
因此再往后退一步,退到了状态:1 __ __。第二个空位上除了填过的 2,3,没有其他数字可以填。
因此再往后退一步,退到了状态:__ __ __。第一个空位上除了填过的 1,还可以填 2。第一个空位上填写 2,填写后为:2 __ __
填好第一个空位,填第二个空位,第二个空位可以填 1,填写后为:2 1 __
填好第二个空位,填第三个空位,第三个空位可以填 3,填写后为:2 1 3
这时候,空位填完,无法继续填数,所以这是一种方案,输出。
然后往后退一步,退到了状态:2 1 __ 。剩余第三个空位没有填数。第三个空位上除了填过的 3,没有其他数字可以填。
因此再往后退一步,退到了状态:2 __ __。第二个空位上除了填过的 1,还可以填 3。第二个空位上填写 3,填写后为:2 3 __
填好第二个空位,填第三个空位,第三个空位可以填 1,填写后为:2 3 1
这时候,空位填完,无法继续填数,所以这是一种方案,输出。
然后往后退一步,退到了状态:2 3 __ 。剩余第三个空位没有填数。第三个空位上除了填过的 1,没有其他数字可以填。
因此再往后退一步,退到了状态:2 __ __。第二个空位上除了填过的 1,3,没有其他数字可以填。
因此再往后退一步,退到了状态:__ __ __。第一个空位上除了填过的 1,2,还可以填 3。第一个空位上填写 3,填写后为:3 __ __
填好第一个空位,填第二个空位,第二个空位可以填 1,填写后为:3 1 __
填好第二个空位,填第三个空位,第三个空位可以填 2,填写后为:3 1 2
这时候,空位填完,无法继续填数,所以这是一种方案,输出。
然后往后退一步,退到了状态:3 1 __ 。剩余第三个空位没有填数。第三个空位上除了填过的 2,没有其他数字可以填。
因此再往后退一步,退到了状态:3 __ __。第二个空位上除了填过的 1,还可以填 2。第二个空位上填写 2,填写后为:3 2 __
填好第二个空位,填第三个空位,第三个空位可以填 1,填写后为:3 2 1
这时候,空位填完,无法继续填数,所以这是一种方案,输出。
然后往后退一步,退到了状态:3 2 __ 。剩余第三个空位没有填数。第三个空位上除了填过的 1,2,没有其他数字可以填。
因此再往后退一步,退到了状态:3 __ __。第二个空位上除了填过的 1,2,没有其他数字可以填。
因此再往后退一步,退到了状态:__ __ __。第一个空位上除了填过的 1,2,3,没有其他数字可以填。
算法:
代码:
#include
#include
#include
using namespace std;
const int N = 10;
int n;
int path[N]; // 从0到n-1共n个位置 存放一个排列
bool st[N];// 存放每个数字的使用状态 true表示使用了 false表示没使用过
void dfs(int u)
{
if(u==n)//数字填完了,输出
{
for(int i=0;i<n;i++)
printf("%d ",path[i]);
cout << endl;
return;
}
for(int i=1;i<=n;i++)//空位上可以选择的数字为:1 ~ n
{
if(!st[i])//如果数字 i 没有被用过
{
path[u]=i;// 把 i 填入数字排列的位置上
st[i]=true;// 表示该数字用过了 不能再用
dfs(u+1);// 填下一位
st[i]=false;// 恢复现场 该数字后续可用
}
}
}
int main()
{
cin >> n;
dfs(0);
return 0;
}
对角线 dg[u+i]
,反对角线udg[n−u+i]
中的下标 u+i
和 n−u+i
表示的是截距
下面分析中的(x,y)
相当于上面的(u,i)
y=x+b
, 截距 b=y−x
,因为我们要把 b
当做数组下标来用,显然 b 不能是负的,所以我们加上 +n
(实际上+n+4,+2n都行),来保证是结果是正的,即 y - x + n
y=−x+b
, 截距是 b=y+x
,这里截距一定是正的,所以不需要加偏移量核心目的: 找一些合法的下标来表示dg
或udg
是否被标记过,所以如果你愿意,你取 udg[n+n−u+i] 也可以,只要所有(u,i)
对可以映射过去就行
代码:
#include
#include
#include
using namespace std;
const int N = 20;
int n;
char g[N][N];
int col[N],dg[N],udg[N];
void dfs(int u)
{
if(u==n)
{
for(int i=0;i<n;i++)
puts(g[i]);
puts("");
}
for(int i=0;i<n;i++)
{
if(!col[i]&&!dg[i+u]&&!udg[i-u+n])
{
g[u][i]='Q';
col[i]=dg[i+u]=udg[i-u+n]=true;
dfs(u+1);
col[i]=dg[u+i]=udg[n-u+i]=false;
g[u][i]='.';
}
}
}
int main()
{
cin >> n;
for(int i=0;i<n;i++)
for(int j=0;j<n;j++)
g[i][j]='.';
dfs(0);
return 0;
}
DFS按每个元素枚举
每个位置都有两种情况,总共有 n^2
个位置
代码:
#include
#include
#include
using namespace std;
const int N = 20;
char g[N][N];
int col[N],row[N],dg[N],udg[N];// 因为是一个个搜索,所以加了row
int n;
void dfs(int x,int y,int s)// s表示已经放上去的皇后个数
{
if(y==n) x++,y=0;// 处理超出边界的情况
if(x==n)// x==n说明已经枚举完n^2个位置了
{
if(s==n)// s==n说明成功放上去了n个皇后
{
for(int i=0;i<n;i++) puts(g[i]);
puts("");
}
return;
}
//不放皇后
dfs(x,y+1,s);
//放皇后
if(!col[x]&&!row[y]&&!dg[y-x+n]&&!udg[y+x])
{
g[x][y]='Q';
col[x]=row[y]=dg[y-x+n]=udg[y+x]=true;
dfs(x,y+1,s+1);
col[x]=row[y]=dg[y-x+n]=udg[y+x]=false;
g[x][y]='.';
}
}
int main()
{
cin >> n;
for(int i=0;i<n;i++)
for(int j=0;j<n;j++)
g[i][j]='.';
dfs(0,0,0);
return 0;
}
算法思想:
只能保证搜到,不能保证最短。
相对BFS,DFS(用系统栈)代码短一些。
代码:
#include
#include
#include
using namespace std;
const int N = 110;
char g[N][N];
bool st[N][N];
int k,n;
int xa,ya,xb,yb;
bool dfs(int x,int y)
{
if(g[x][y]=='#') return false;
if(x==xb&&y==yb) return true;
st[x][y]=true;
int dx[4] = {-1, 0, 1, 0}, dy[4] = {0, 1, 0, -1};
for(int i=0;i<4;i++)
{
int a=x+dx[i],b=y+dy[i];
if(a>=0&&a<n&&b>=0&&b<n&&!st[a][b])
{
if(dfs(a,b)) return true;
}
}
return false;
}
int main()
{
cin >> k;
while(k--)
{
cin >> n;
for(int i=0;i<n;i++) cin >> g[i];
memset(st,false,sizeof st);
cin >> xa >> ya >> xb >> yb;
if(dfs(xa,ya)) puts("YES");
else puts("NO");
}
return 0;
}
算法思想:
每一个格子只会被搜一次,不需要恢复现场。
若将整个棋盘当成一个状态,进行一次染色,需要恢复现场。
关键:一个状态需要多次使用时才会恢复现场。
深搜代码:
#include
#include
#include
using namespace std;
const int N = 30;
int n,m;
char g[N][N];
bool st[N][N];
int dfs(int x,int y)
{
int cnt=1;
st[x][y]=true;
int dx[4] = {-1, 0, 1, 0}, dy[4] = {0, 1, 0, -1};
for(int i=0;i<4;i++)
{
int a=x+dx[i],b=y+dy[i];
if(a>=0&&a<n&&b>=0&&b<m&&g[a][b]=='.'&&!st[a][b])
{
cnt+=dfs(a,b);
}
}
return cnt;
}
int main()
{
while(cin>>m >>n,n||m)
{
for(int i=0;i<n;i++) cin >> g[i];
memset(st,false,sizeof st);
for(int i=0;i<n;i++)
for(int j=0;j<m;j++)
if(g[i][j]=='@') cout << dfs(i,j) << endl;
}
}
1.宽搜代码:
#include
#include
#include
#include
#define x first
#define y second
using namespace std;
typedef pair<int, int> PII;
const int N = 100;
int n,m;
char g[N][N];
bool st[N][N];
int bfs(PII start)
{
queue<PII> q;
q.push({start});
memset(st,false,sizeof st);
st[start.x][start.y]=true;
int cnt=0;
while(q.size())
{
cnt++;
auto t=q.front();
q.pop();
int dx[4] = {-1, 0, 1, 0}, dy[4] = {0, 1, 0, -1};
for(int i=0;i<4;i++)
{
int a=t.x+dx[i],b=t.y+dy[i];
if(a>=0&&a<n&&b>=0&&b<m&&g[a][b]=='.'&&!st[a][b])
{
q.push({a,b});
st[a][b]=true;
}
}
}
return cnt;
}
int main()
{
while(scanf("%d %d",&m,&n),n||m)
{
for(int i=0;i<n;i++) cin >> g[i];
PII start;
for(int i=0;i<n;i++)
for(int j=0;j<m;j++)
if(g[i][j]=='@') start={i,j};
cout << bfs(start) << endl;
}
return 0;
}
想一个搜索顺序,将所有情况搜索到。
注意:恢复现场
代码:
#include
#include
#include
using namespace std;
const int N = 20;
int n,m;
int ans;
bool st[N][N];
void dfs(int x,int y,int cnt)
{
if(cnt==n*m)
{
ans++;
return;
}
st[x][y]=true;
int dx[8] = {-2, -1, 1, 2, 2, 1, -1, -2};
int dy[8] = {1, 2, 2, 1, -1, -2, -2, -1};
for(int i=0;i<8;i++)
{
int a=x+dx[i],b=y+dy[i];
if(a>=0&&a<n&&b>=0&&b<m&&!st[a][b])
{
dfs(a,b,cnt+1);
}
}
st[x][y]=false;
}
int main()
{
int T;
cin >> T;
while(T--)
{
int x,y;
cin >> n >> m >> x >>y;
ans=0;
dfs(x,y,1);
printf("%d\n",ans);
}
}
算法思想:
首先枚举以给定字母开头的单词,对于每个单词分别向下递归搜索,每次枚举可以接在后面的所有单词,直到不能搜为止,计算长度最大值。
代码:
#include
#include
#include
using namespace std;
const int N = 30;
int g[N][N];
string word[N];
int used[N];
int n;
int ans;
void dfs(string dragon,int last)
{
ans=max((int)dragon.size(),ans);
used[last]++;
for(int i=0;i<n;i++)
{
if(g[last][i] && used[i]<2)
dfs(dragon+word[i].substr(g[last][i]),i);
}
used[last]--;
}
int main()
{
cin >> n;
for(int i=0;i<n;i++) cin >> word[i];
char start;
cin >> start;
for(int i=0;i<n;i++)
for(int j=0;j<n;j++)
{
string a=word[i],b=word[j];
for(int k=1;k<min(a.size(),b.size());k++)
{
if(a.substr(a.size()-k,k)==b.substr(0,k))
{
g[i][j]=k;
break;
}
}
}
for(int i=0;i<n;i++)
if(word[i][0]==start)
{
dfs(word[i],i);
}
cout << ans << endl;
return 0;
}
算法思想:
如果两个数互质,连一条边,表示有关系。将图中的点最少分成多少组,可以使得每一组内没有边。转化为图论。
代码:
未来不是我们想要的,可以提前退出搜索,提高搜索效率。
常见的剪枝策略:
1.优化搜索顺序,大部分情况下,我们应该优先搜索分支较少的节点。
2.排除等效冗余。不考虑的情况下,用组合的方式进行搜索。
3.可行性剪枝
4.最优性剪枝
5.记忆化搜索(DP)
代码:
每次取出队头元素,将该元素拓展出的所有元素放到队尾。取出的顺序是层次遍历的顺序,加入的顺序也是层次遍历的顺序。
当一个图中所有边的权重为1,可以考虑用宽搜
BFS: 求最小/最短
,基于迭代的算法
,不会爆栈。
数据结构:判重数组st[ ] (入队时判重,出队时判重),queue
//宽搜框架
//二维 typedef pair PII; #define x first PII start,end;
//三维 struct Point{int x,y,z;};
queue<-初始状态
queue<PII> q;q.push(start);
int hh=0,tt=0;q[0]=start;q[0]={0,0};
//dist[]数组初始化
memset(dist,-1,sizeof dist);
dist[start.x][start.y]=0;
//判重数组初始化
st[0][0]=true;
st[start.x][start.y]=true;//表明该数已经用过
//一些多组输入的题目,需要进行初始化操作
memset(st,false,sizeof st);
while(queue非空)//q.size() hh<=tt
{
t<-队头//auto t=q.front();q.pop(); auto t=q[hh++];
//定义方向,二维dx、dy,三维dx、dy、dz
for(拓展t)//int i=0;i<4/6;i++
{
var<-t拓展的新节点
//int x=t.x+dx[i],y=t.y+dy[i];
if(!st[var])
var->队尾
//判断
if(是否在范围内&&dist[x][y]==-1&&g[x][y]!='#') 未拓展且不是障碍
//更新dist
dist[x][y]=dist[t.x][t.y]+1;
//判断是否结束
if(g[x][y]=='E') return dist[x][y];
if(end==make_pair(x,y)) return dist[x][y];
if(x==end.x&&y==end.y&&z==end.z) return dist[x][y][z];
//拓展新的节点
q.push({x,y}) q[++tt]={x,y};
}
}
//return -1;
常见的题型
求最短距离,引入dist[][]
,初始化为-1
,起点的dist
值为0
,当满足条件时,将dist[a][b]=dist[t.x][t.y]+1
,并判断是否到达终点,返回dist[a][b]
dist[][]作用
:一般初始化为-1
。既可以判重,又可以记录答案,即距离。
Flood Fill,引入判重数组st[][]
,将扫描的点设置为st[][]=true
,进行条件判断若!st[a][b]
,还没有扫描,执行相应的操作,并设置st[a][b]=true
用a[N][N]
接收地图,dist[N][N]
存储到每个点的路径长度。
从起点出发,广度优先遍历地图:
代码:
#include
#include
#include
#include
#define x first
#define y second
using namespace std;
typedef pair<int, int> PII;
const int N = 210;
int dx[4]={-1,0,1,0},dy[4]={0,1,0,-1};
int n,m;
char g[N][N];
int dist[N][N];
int bfs(PII start,PII end)
{
queue<PII> q;
q.push(start);//队列初始化
memset(dist,-1,sizeof dist);
dist[start.x][start.y]=0;
while(q.size())//队列非空
{
auto t=q.front();//取队头元素
q.pop();//将队头元素出队列
//进行t拓展,循环往上下左右四个方向走
for(int i=0;i<4;i++)
{
int x=t.x+dx[i],y=t.y+dy[i];
//判断当前节点是否能继续走
if (x>=0&&x<n&&y>=0&&y<m&&g[x][y]!='#'&&dist[x][y]==-1)
{
dist[x][y] = dist[t.x][t.y] + 1;//更新
if (g[x][y] == 'E') return dist[x][y];
q.push({x, y});
}
}
}
return -1;
}
int main()
{
int T;
cin >> T;
while(T--)
{
cin >> n >> m;
for(int i=0;i<n;i++) scanf("%s",g[i]);
PII start,end;
for(int i=0;i<n;i++)
for(int j=0;j<m;j++)
if(g[i][j]=='S') start={i,j};
else if(g[i][j]=='E') end={i,j};
int distance = bfs(start,end);
if(distance == -1) puts("oop!");
else printf("%d\n",distance);
}
return 0;
}
#include
#include
#include
#include
#define x first
#define y second
using namespace std;
typedef pair<int, int> PII;
const int N = 110;
int dx[4] = {-1, 0, 1, 0}, dy[4] = {0, 1, 0, -1};
int n,m;
int g[N][N];
int dist[N][N];
int bfs(int x,int y)
{
queue<PII> q;
q.push({x,y});
memset(dist,-1,sizeof dist);
dist[x][y]=0;
while(q.size())
{
auto t=q.front();
q.pop();
for(int i=0;i<4;i++)
{
int x=t.x+dx[i],y=t.y+dy[i];
if(x>=1&&x<=n&&y>=1&&y<=m&&g[x][y]==0&&dist[x][y]==-1)
{
dist[x][y]=dist[t.x][t.y]+1;
if(x==n&&y==m) return dist[x][y];
q.push({x,y});
}
}
}
}
int main()
{
cin >> n >> m;
for(int i=1;i<=n;i++)
for(int j=1;j<=m;j++)
cin >> g[i][j];
int res=bfs(1,1);
cout << res << endl;
return 0;
}
输出路径
二维拓展为三维:
沿着x轴前进:1,0,0
沿着x轴后退:-1,0,0
沿着y轴前进:0,1,0
沿着y轴后退:0,-1,0
沿着z轴前进:0,0,1
沿着z轴后退:0,0,-1
代码:
#include
#include
#include
#include
using namespace std;
const int N = 110;
struct Point
{
int x,y,z;
};
Point q[N*N*N];
char g[N][N][N];
int dist[N][N][N];
int L,R,C;
int dx[6]={1,-1,0,0,0,0};
int dy[6]={0,0,1,-1,0,0};
int dz[6]={0,0,0,0,1,-1};
int bfs(Point start,Point end)
{
int hh=0,tt=0;
q[0]=start;
memset(dist,-1,sizeof dist);
dist[start.x][start.y][start.z]=0;
while(hh<=tt)
{
auto t=q[hh++];
for(int i=0;i<6;i++)
{
int x=t.x+dx[i],y=t.y+dy[i],z=t.z+dz[i];
if(x>=0&&x<L&&y>=0&&y<R&&z>=0&&z<C&&g[x][y][z]!='#'&&dist[x][y][z]==-1)
{
dist[x][y][z]=dist[t.x][t.y][t.z]+1;
if(x==end.x&&y==end.y&&z==end.z) return dist[x][y][z];
q[++tt]={x,y,z};
}
}
}
return -1;
}
int main()
{
Point start,end;
while(scanf("%d %d %d",&L,&R,&C),L||R||C)
{
for(int i=0;i<L;i++)
for(int j=0;j<R;j++)
{
scanf("%s",g[i][j]);
for(int k=0;k<C;k++)
if(g[i][j][k]=='S') start={i,j,k};
else if(g[i][j][k]=='E') end={i,j,k};
}
int distance=bfs(start,end);
if(distance==-1) puts("Trapped!");
else printf("Escaped in %d minute(s).\n",distance);
}
return 0;
}
从一个起点开始,每一次选择一个新加进来的格子,直到不能拓展新的格子为止。
可以在线性时间复杂度内,找到某个点所在的连通块。
算法思想:
遍历每一个格子,对于当前格子如果是水,即W,进行cnt++,执行flood fill。
需要注意的在st[x][y]=true
,表示将该点放到判重数组,为1表示已经用过。
代码:
#include
#include
#include
#include
#define x first
#define y second
using namespace std;
const int N = 1010;
typedef pair<int, int> PII;
int n,m;
char g[N][N];
bool st[N][N];
void bfs(int sx,int sy)
{
queue<PII> q;
q.push({sx,sy});
st[sx][sy]=true;//表明已经遍历过了
while(q.size())
{
auto t=q.front();
q.pop();
//8方向遍历
int dx[8] = {-1, -1, -1, 0, 1, 1, 1, 0};
int dy[8] = {-1, 0, 1, 1, 1, 0, -1, -1};
for(int i=0;i<8;i++)
{
int x=t.x+dx[i],y=t.y+dy[i];
if(x>=0&&x<n&&y>=0&&y<m&&g[x][y]=='W'&&!st[x][y])
{
q.push({x,y});
st[x][y]=true;
}
}
}
}
int main()
{
scanf("%d %d",&n,&m);
for(int i=0;i<n;i++) scanf("%s",g[i]);
int cnt=0;
for(int i=0;i<n;i++)
for(int j=0;j<m;j++)
if(g[i][j]=='W'&&!st[i][j])
{
bfs(i,j);
cnt++;
}
cout << cnt << endl;
}
代码:
#include
#include
#include
#define x first
#define y second
#include
const int N = 55;
using namespace std;
typedef pair<int, int> PII;
int n,m;
int g[N][N];
bool st[N][N];
int bfs(int sx,int sy)
{
queue<PII> q;
int area=0;
q.push({sx,sy});
st[sx][sy]=true;
//按照西北东南四个方向 (0,-1) (-1,0) (0,1) (1,0)
int dx[4]={0,-1,0,1},dy[4]={-1,0,1,0};
while(q.size())
{
auto t=q.front();
area++;
q.pop();
for(int i=0;i<4;i++)
{
int x=t.x+dx[i],y=t.y+dy[i];
if(x>=0&&x<n&&y>=0&&y<m&&!st[x][y]&&!(g[t.x][t.y]>>i&1))//若g[t.x][t.y]>>i&1为1,表示该方向有墙
{
q.push({x,y});
st[x][y]=true;
}
}
}
return area;
}
int main()
{
cin >> n >> m;
for(int i=0;i<n;i++)
for(int j=0;j<m;j++)
cin >> g[i][j];
int cnt=0,area=0;
for(int i=0;i<n;i++)
for(int j=0;j<m;j++)
if(!st[i][j])
{
area=max(area,bfs(i,j));
cnt++;
}
cout << cnt << endl << area << endl;
return 0;
}
往外进行拓展的时候,判断一下拓展的格子是否属于连通块,并统计两者之间的关系
找到高度一致的连通块,若该连通块周围
统计连通块(边界)与其相邻节点的关系:
代码:
#include
#include
#include
#include
#define x first
#define y second
using namespace std;
typedef pair<int, int> PII;
const int N = 1010;
int n;
int h[N][N];
bool st[N][N];
void bfs(int sx,int sy,bool &has_higher,bool &has_lower)
{
queue<PII> q;
q.push({sx,sy});
st[sx][sy]=true;
while(q.size())
{
auto t=q.front();
q.pop();
int dx[8] = {-1, -1, -1, 0, 1, 1, 1, 0};
int dy[8] = {-1, 0, 1, 1, 1, 0, -1, -1};
for(int i=0;i<8;i++)
{
int x=t.x+dx[i],y=t.y+dy[i];
if(x>=0&&x<n&&y>=0&&y<n)
{
if(h[x][y]!=h[t.x][t.y])//山脉边界
{
if(h[x][y]>h[t.x][t.y]) has_higher=true;
else has_lower=true;
}
else if(!st[x][y])//相等,只有没有遍历过的时候才进行遍历
{
q.push({x,y});
st[x][y]=true;
}
}
}
}
}
int main()
{
cin >> n;
for(int i=0;i<n;i++)
for(int j=0;j<n;j++)
cin >> h[i][j];
int peak=0,valley=0;
for(int i=0;i<n;i++)
for(int j=0;j<n;j++)
if(!st[i][j])
{
bool has_higher=false,has_lower=false;
bfs(i,j,has_higher,has_lower);
if(!has_higher) peak++;
if(!has_lower) valley++;
}
cout << peak << " " << valley << endl;
}
PII类型的pre[][]
数组即用来判重标记,又存储当前路径
代码:
#include
#include
#include
#include
#define x first
#define y second
using namespace std;
typedef pair<int, int> PII;
const int N = 1010;
int g[N][N];
int n;
PII pre[N][N];
void bfs(int sx,int sy)
{
queue<PII> q;
q.push({sx,sy});
memset(pre,-1,sizeof pre);
pre[sx][sy]={0,0};
while(q.size())
{
auto t=q.front();
q.pop();
int dx[4] = {-1, 0, 1, 0}, dy[4] = {0, 1, 0, -1};
for(int i=0;i<4;i++)
{
int a=t.x+dx[i],b=t.y+dy[i];
if(a>=0&&a<n&&b>=0&&b<n&&g[a][b]==0)
{
if(pre[a][b].x!=-1) continue;
q.push({a,b});
pre[a][b]=t;
}
}
}
}
int main()
{
cin >> n;
for(int i=0;i<n;i++)
for(int j=0;j<n;j++)
scanf("%d",&g[i][j]);
bfs(n-1,n-1);
PII end(0,0);
while(true)
{
printf("%d %d\n",end.x,end.y);
if(end.x==n-1&&end.y==n-1) break;
end=pre[end.x][end.y];
}
return 0;
}
用st[][]
数组进行标记
#include
#include
#include
#include
#define x first
#define y second
using namespace std;
typedef pair<int, int> PII;
const int N = 1010;
int g[N][N];
int n;
PII pre[N][N];
bool st[N][N];
void bfs(int sx,int sy)
{
queue<PII> q;
q.push({sx,sy});
st[sx][sy]=true;
while(q.size())
{
auto t=q.front();
q.pop();
int dx[4] = {-1, 0, 1, 0}, dy[4] = {0, 1, 0, -1};
for(int i=0;i<4;i++)
{
int a=t.x+dx[i],b=t.y+dy[i];
if(a>=0&&a<n&&b>=0&&b<n&&g[a][b]==0&&!st[a][b])
{
q.push({a,b});
st[a][b]=true;
pre[a][b]=t;
}
}
}
}
int main()
{
cin >> n;
for(int i=0;i<n;i++)
for(int j=0;j<n;j++)
scanf("%d",&g[i][j]);
bfs(n-1,n-1);
PII end(0,0);
while(true)
{
printf("%d %d\n",end.x,end.y);
if(end.x==n-1&&end.y==n-1) break;
end=pre[end.x][end.y];
}
return 0;
}
代码:
#include
#include
#include
#include
#define x first
#define y second
using namespace std;
typedef pair<int, int> PII;
int dx[8] = {-2, -1, 1, 2, 2, 1, -1, -2};
int dy[8] = {1, 2, 2, 1, -1, -2, -2, -1};
const int N = 200;
char g[N][N];
int n,m;
int dist[N][N];
int bfs(PII start)
{
queue<PII> q;
q.push(start);
memset(dist,-1,sizeof dist);
dist[start.x][start.y]=0;
while(q.size())
{
auto t=q.front();
q.pop();
for(int i=0;i<8;i++)
{
int a=t.x+dx[i],b=t.y+dy[i];
if(a>=0 && a<n && b>=0 &&b<m && g[a][b]!='*' && dist[a][b]==-1)
{
dist[a][b]=dist[t.x][t.y]+1;
if(g[a][b]=='H') return dist[a][b];
q.push({a,b});
}
}
}
return -1;
}
int main()
{
cin >> m >> n;
for(int i=0;i<n;i++)
cin >> g[i];
PII start;
for(int i=0;i<n;i++)
for(int j=0;j<m;j++)
if(g[i][j]=='K') start={i,j};
cout << bfs(start) << endl;
return 0;
}
代码:
#include
#include
#include
using namespace std;
const int N = 200010;
int n,k;
int dist[N];
int q[N];
int bfs()
{
memset(dist,-1,sizeof dist);
dist[n]=0;
q[0]=n;//queue q;q.push(n);
int hh=0,tt=0;
while(hh<=tt)
{
int t=q[hh++];//int t=q.front();q.pop();
if(t==k) return dist[k];
if(t+1<N && dist[t+1]==-1)
{
dist[t+1]=dist[t]+1;
q[++tt]=t+1;//q.push(t+1);
}
if(t-1>=0&& dist[t-1]==-1)
{
dist[t-1]=dist[t]+1;
q[++tt]=t-1;
}
if(t*2<N&& dist[2*t]==-1)
{
dist[2*t]=dist[t]+1;
q[++tt]=2*t;
}
}
return -1;
}
int main()
{
cin >> n >> k;
cout << bfs() << endl;
return 0;
}
算法思想:
有很多起点,每次求出某点到离它最近的起点的距离。
代码:
#include
#include
#include
#include
#define x first
#define y second
using namespace std;
typedef pair<int, int> PII;
const int N = 1010;
int n,m;
char g[N][N];
int dist[N][N];
void bfs()
{
queue<PII> q;
memset(dist,-1,sizeof dist);
for(int i=0;i<n;i++)
for(int j=0;j<m;j++)
if(g[i][j]=='1')
{
dist[i][j]=0;
q.push({i,j});
}
while(q.size())
{
auto t=q.front();
q.pop();
int dx[4] = {-1, 0, 1, 0}, dy[4] = {0, 1, 0, -1};
for(int i=0;i<4;i++)
{
int x=t.x+dx[i],y=t.y+dy[i];
if(x>=0&&x<n&&y>=0&&y<m&&dist[x][y]==-1)
{
dist[x][y]=dist[t.x][t.y]+1;
q.push({x,y});
}
}
}
}
int main()
{
cin >> n >> m;
for(int i=0;i<n;i++) cin >> g[i];
bfs();
for(int i=0;i<n;i++)
{
for(int j=0;j<m;j++)
{
printf("%d ",dist[i][j]);
}
cout << endl;
}
return 0;
}
有向图中的最短距离
1.如何存储状态? 哈希存储。康拓展开
map、unordered-map
将答案1、2、3、4、5、6、7、8放到队列队头,用宽搜搜索所有的状态,直到搜到终点为止。每次拓展时,进行A、B、C三种操作,得到一个字符串判断是否搜到过,未搜,存队列。
2.如何存方案? 存储每一个状态由哪个状态转移过来的。终点->起点
3.如何得到字典序最小的操作? 往队列插的时候,按照先A再B再C的顺序插入,一定可以得到一个最小字典序的答案。
代码:
#include
#include
#include
#include
#include
#define x first
#define y second
using namespace std;
char g[2][4];
unordered_map<string,int> dist;
unordered_map<string,pair<char,string>> pre;
void set(string state)
{
for(int i=0;i<4;i++) g[0][i]=state[i];
for(int i=3,j=4;i>=0;i--,j++) g[1][i]=state[j];
}
string get()
{
string res;
for(int i=0;i<4;i++) res+=g[0][i];
for(int i=3;i>=0;i--) res+=g[1][i];
return res;
}
string move0(string state)
{
set(state);
for(int i=0;i<4;i++) swap(g[0][i],g[1][i]);
return get();
}
string move1(string state)
{
set(state);
char v0=g[0][3],v1=g[1][3];
for(int i=3;i>0;i--)
for(int j=0;j<2;j++)
g[j][i]=g[j][i-1];
g[0][0]=v0,g[1][0]=v1;
return get();
}
string move2(string state)
{
set(state);
char v=g[0][1];
g[0][1]=g[1][1];
g[1][1]=g[1][2];
g[1][2]=g[0][2];
g[0][2]=v;
return get();
}
void bfs(string start,string end)
{
if(start==end) return;
queue<string> q;
q.push(start);
dist[start]=0;
while(q.size())
{
auto t=q.front();
q.pop();
string m[3];
m[0]=move0(t);
m[1]=move1(t);
m[2]=move2(t);
for(int i=0;i<3;i++)
{
string str=m[i];
if(dist.count(str)==0)
{
dist[str]=dist[t]+1;
pre[str]={char(i+'A'),t};
if(str==end) break;
q.push(str);
}
}
}
}
int main()
{
int x;
string start,end;
for(int i=0;i<8;i++)
{
cin >> x;
end+=char(x+'0');
}
for(int i=0;i<8;i++) start+=char(i+'1');
bfs(start,end);
cout << dist[end] << endl;
string res;
while(end!=start)
{
res+=pre[end].x;
end=pre[end].y;
}
reverse(res.begin(),res.end());
if(res.size()) cout << res << endl;
return 0;
}
与一般的BFS的差别: 如果拓展出来的边的权重是0,插入到队头,拓展出来的边的权重是1,插入到队尾。
解决问题:
双端队列
主要解决图中边的权值只有0
或者1
的最短路问题
操作:
每次从队头取出元素,并进行拓展其他元素时
0
,则将该元素插入到队头1
,则将该元素插入到队尾与堆优化Dijkstra 一样,必须在出队时才知道每个点最终的最小值,而和一般的bfs不一样,原因是如下图所示
首先明确的一点是,这里是图中的格子和点是不一样的,点是格子上的角角上的点,每个点都有4个方向可以走,分别对应的是左上角,右上角,右下角,左下角,
踩过格子到达想去的点时,需要判断是否需要旋转电线,若旋转电线表示从 当前点 到 想去的点 的边权是1
,若不旋转电线则边权是0
按左上角,右上角,右下角,左下角遍历的顺序
dx[]
和dy[]
表示可以去其他点的方向id[]
和iy[]
表示需要踩某个方向的各种才能去到相应的点cs[]
表示当前点走到4
个方向的点理想状态下格子形状(边权是0
的状态)代码:
求斐波那契数列, f ( n ) = f ( n − 1 ) + f ( n − 2 ) , ( n ≥ 3 ) f(n)=f(n-1)+f(n-2),(n≥3) f(n)=f(n−1)+f(n−2),(n≥3)
//实现斐波那契额数列
int fun(int n)
{
if(n==1) return 1;
if(n==2) return 2;
else return f(n-1)+f(n-2);
}
算法思想:
递归,dfs,最重要的是顺序。从1~n依次考虑每个数选或者不选。
代码:
#include
#include
#include
using namespace std;
const int N = 16;
int st[N];//用来记录每个位置当前的状态,0表示还没有考虑,1表示选当前数,2表示不选当前数
int n;
void dfs(int u)
{
if(u>n)
{
for(int i=1;i<=n;i++)
if(st[i]==1)
printf("%d ",i);
cout << endl;
return ;
}
st[u]=2;
dfs(u+1);//第一个分支,不选
st[u]=0;//恢复现场
st[u]=1;
dfs(u+1);//第二个分支,选
st[u]=0;//恢复现场
}
int main()
{
cin >> n;
dfs(1);
return 0;
}
STL的做法代码:
#include
using namespace std;
int n;
vector<int> num;
void dfs(int k)
{
//到达枚举边界,输出结果并结束
if(k == n + 1)
{
for(int i = 0;i < num.size();++i)
cout << num[i] << " ";
cout << endl;
return;
}
//不选择这个数
dfs(k+1);
//选择这个数
num.push_back(k);
dfs(k+1);
//回溯
num.pop_back();
}
int main(void)
{
cin >> n;
dfs(1);
return 0;
}
算法思想:
顺序1:依次枚举每个数放到哪个位置
顺序2:依次枚举每个位置放哪个数
代码:
#include
#include
#include
using namespace std;
const int N = 10;
int n;
bool used[N];//false表示还没用过,true表示已经用过
int path[N];
void dfs(int u)
{
if(u==n)
{
for(int i=0;i<n;i++)
printf("%d ",path[i]);
cout << endl;
return;
}
//依次枚举每个分支,即当前位置可以填哪些数
for(int i=1;i<=n;i++)
{
if(!used[i])
{
path[u]=i;
used[i]=true;
dfs(u+1);
//恢复现场
used[i]=false;
}
}
}
int main()
{
cin >> n;
dfs(0);
return 0;
}
算法思想:
在排列型枚举的基础上加一些限定,局部考虑,只需要保证每次新加的数大于前一个数。
代码:
#include
#include
#include
using namespace std;
const int N = 26;
int n,m;
int path[N];
bool used[N];
int start;
void dfs(int u,int start)
{
if(u+n-start<m) //正在选定第u个数,已经选了u-1个数,加上后面选的数若不够m,剪枝
return ;
if(u==m+1)
{
for(int i=1;i<=m;i++)
printf("%d ",path[i]);
cout << endl;
return ;
}
for(int i=start;i<=n;i++)
{
path[u]=i;
dfs(u+1,i+1);
path[u]=0; //恢复现场
}
}
int main()
{
cin >> n >> m;
dfs(1,1);
return 0;
}
动态数组做法代码:
#include
using namespace std;
int n,m;
vector<int> num;
void dfs(int k)
{
//如题解所述
if(num.size() > m || num.size() + (n - k + 1) < m)
return;
//到达枚举边界,输出结果并结束
if(k == n + 1)
{
for(int i = 0;i < num.size();++i)
cout << num[i] << " ";
cout << endl;
return;
}
//选择这个数
num.push_back(k);
dfs(k+1);
//回溯
num.pop_back();
//不选择这个数
dfs(k+1);
}
int main(void)
{
cin >> n >> m;
dfs(1);
return 0;
}
算法思想:
递归枚举全排列
枚举a、b的位数
判断等式是否成立
代码:
有向图、无向图
有向图的存储:
邻接矩阵(用的少,二维数组g[a][b]
)、邻接表(用的多,单链表)