目录
1 flood fill
1.Lake Counting
信息学奥赛一本通(C++版)在线评测系统 (ssoier.cn)
#include<bits/stdc++.h>
using namespace std;
#define x first
#define y second
typedef pair<int,int> pii;
const int N=115;
int n,m,cnt=0;
char g[N][N];
bool st[N][N];
pii q[N*N];
void bfs(int sx,int sy)
{
int hh=0,tt=0;
q[0]={sx,sy};
st[sx][sy]=true;
while(hh<=tt)
{
auto t=q[hh++];
for(int i=t.x-1;i<=t.x+1;i++)//因为求八个周围的,则循环枚举两层即可
for(int j=t.y-1;j<=t.y+1;j++)
{
if(i<0||i>=n||j<0||j>=m) continue;//假如越界则continue
if(st[i][j]) continue;//假如标记过则continue
if(g[i][j]=='.') continue;//假如是干燥的则continue
st[i][j]=true;//标记这个点被标记过
q[++tt]={i,j};//把这个积水放进队列用,以后用他来更新其他的积水
}
}
}
int main()
{
cin>>n>>m;
for(int i=0;i<n;i++) cin>>g[i];
for(int i=0;i<n;i++)//遍历每一个点
for(int j=0;j<m;j++)
if(!st[i][j]&&g[i][j]=='W')//假如之前没标记过并且是新的水洼
{
bfs(i,j);//遍历这个水洼覆盖的水洼
cnt++;//水洼数量加一
}
cout<<cnt<<endl;
return 0;
}
2.The Castle
信息学奥赛一本通(C++版)在线评测系统 (ssoier.cn)
#include<bits/stdc++.h>
using namespace std;
#define x first
#define y second
typedef pair<int,int> pii;
const int N=55;
int n,m,cnt=0,ans=0;
int g[N][N];
bool st[N][N];
pii q[N*N];
int dx[4]={0,-1,0,1},dy[4]={-1,0,1,0};//按照题目的方向西北东南的方向
int bfs(int sx,int sy)
{
int hh=0,tt=0;
q[0]={sx,sy};
st[sx][sy]=true;
while(hh<=tt)
{
auto t=q[hh++];//取出队头
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) continue;//假如越界则continue
if(st[a][b]) continue;//假如标记过则continue
if((g[t.x][t.y]>>i)&1) continue;//假如是墙的话则continue
st[a][b]=true;//标记这个点被标记过
q[++tt]={a,b};//把这个点放进队列用,以后用他来更新其他的房间
}
}
return tt+1;//返回房间数量
}
int main()
{
cin>>n>>m;
for(int i=0;i<n;i++)
for(int j=0;j<m;j++)
cin>>g[i][j];
for(int i=0;i<n;i++)//遍历每一个点
for(int j=0;j<m;j++)
if(!st[i][j])//假如之前没标记过,则是新的房间
{
ans=max(ans,bfs(i,j));//更新一遍房间的最大值
cnt++;//水洼数量加一
}
cout<<cnt<<endl;
cout<<ans<<endl;
return 0;
}
3.山峰和山谷
信息学奥赛一本通(C++版)在线评测系统 (ssoier.cn)
#include<bits/stdc++.h>
using namespace std;
#define x first
#define y second
typedef pair<int,int> pii;
const int N=1e3+10;
int n;
int g[N][N];
bool st[N][N];
pii q[N*N];
int bfs(int sx,int sy)
{
int hh=0,tt=0;
bool up=false,low=false;//用来记录是不是山峰或者山谷
st[sx][sy]=true;
q[0]={sx,sy};
while(hh<=tt)
{
auto t=q[hh++];
//因为是要周围八个方向,所以用枚举的方法
for(int i=t.x-1;i<=t.x+1;i++)
for(int j=t.y-1;j<=t.y+1;j++)
{
if(i<0||i>=n||j<0||j>=n) continue;//假如越界
if(g[i][j]>g[t.x][t.y]) low=true;//假如是山谷
else if(g[i][j]<g[t.x][t.y]) up=true;//假如是山峰
else if(!st[i][j])//假如是同一高度且没被标记过的
{
st[i][j]=true;
q[++tt]={i,j};
}
}
}
if(up&&low) return -1;//假如即使山峰并且又是山谷,则啥也不是
else if(up) return 1;//假如山峰
else if(low) return 0;//假如山谷
else return 2;//假如二者都可以
}
int main()
{
int up=0,low=0;
cin>>n;
for(int i=0;i<n;i++)
for(int j=0;j<n;j++)
cin>>g[i][j];
for(int i=0;i<n;i++)
for(int j=0;j<n;j++)
if(!st[i][j])//假如这个点没有被走过
{
int t=bfs(i,j);
if(t==1) up++;//假如是山峰
else if(t==0) low++;//假如是山谷
else if(t==2) up++,low++;//假如即可以作为山峰,又可以作为山谷
}
cout<<up<<" "<<low<<endl;
return 0;
}
2最短路模型
1.迷宫问题
信息学奥赛一本通(C++版)在线评测系统 (ssoier.cn)
#include<bits/stdc++.h>
using namespace std;
#define x first
#define y second
typedef pair<int,int> pii;
const int N=10;
int n;
int g[N][N];
bool st[N][N];//标记那个点被走过了
pii q[N*N],pre[N][N];//pre用来记录某个点的上一步
int dx[4]={-1,0,1,0},dy[4]={0,1,0,-1};
void bfs(int sx,int sy)
{
int hh=0,tt=0;
st[sx][sy]=true;
q[0]={sx,sy};
pre[sx][sy]={0,0};
while(hh<=tt)
{
auto t=q[hh++];
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) continue;//假如越界
if(g[a][b]) continue;//假如是墙,不能走
if(st[a][b]) continue;//假如之前走过这个点
q[++tt]={a,b};
st[a][b]=true;
pre[a][b]=t;//标记这个点由t转移过来的
}
}
}
int main()
{
n=5;
for(int i=0;i<n;i++)
for(int j=0;j<n;j++)
cin>>g[i][j];
bfs(n-1,n-1);//倒序求最短路
pii end(0,0);//正向输出路径
while(1)
{
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;
}
2.武士风度的牛
#include<bits/stdc++.h>
using namespace std;
#define x first
#define y second
typedef pair<int,int> pii;
const int N=160;
int n,m;
char g[N][N];
int dist[N][N];//记录吗,每个点到起点的步数
pii q[N*N];
int dx[]={-1,1,2,2,1,-1,-2,-2},dy[]={2,2,1,-1,-2,-2,-1,1};//题目中的八个方向
int bfs(int sx,int sy,int ex,int ey)
{
int hh=0,tt=0;
q[0]={sx,sy};
memset(dist,-1,sizeof dist);
dist[sx][sy]=0;
while(hh<=tt)
{
auto t=q[hh++];
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) continue;//假如越界
if(g[a][b]=='*') continue;//假如是障碍,不能走
if(dist[a][b]!=-1) continue;//假如之前已经更新过这个点
q[++tt]={a,b};
dist[a][b]=dist[t.x][t.y]+1;//则步数+1
}
}
return dist[ex][ey];//返回到终点的步数
}
int main()
{
cin>>m>>n;
int sx,sy,ex,ey;
for(int i=0;i<n;i++)
for(int j=0;j<m;j++)
{
cin>>g[i][j];
if(g[i][j]=='K') sx=i,sy=j;//标记起点
if(g[i][j]=='H') ex=i,ey=j;//标记终点
}
cout<<bfs(sx,sy,ex,ey)<<endl;//输出最小步数
return 0;
}
3.抓住那头牛
信息学奥赛一本通(C++版)在线评测系统 (ssoier.cn)
#include<bits/stdc++.h>
using namespace std;
const int N=2e5+10;
int n,k;
int dist[N];//记录每个点到起点的步数
int q[N];
int bfs()
{
int hh=0,tt=0;
q[0]=n;
memset(dist,-1,sizeof dist);
dist[n]=0;
while(hh<=tt)
{
int t=q[hh++];
if(t==k) return dist[k];//返回到起点的步数
if(t+1<N&&dist[t+1]==-1)//第一种情况走,且该位置没有被走过
{
dist[t+1]=dist[t]+1;//则该点距离+1
q[++tt]=t+1;//将该点放进队列中
}
if(t-1>=0&&dist[t-1]==-1)//第二种情况走,且该位置没有被走过
{
dist[t-1]=dist[t]+1;//则该点距离+1
q[++tt]=t-1;//将该点放进队列中
}
if(2*t<N&&dist[2*t]==-1)//第三种情况走,且该位置没有被走过
{
dist[2*t]=dist[t]+1;//则该点距离+1
q[++tt]=2*t;//将该点放进队列中
}
}
return 0;
}
int main()
{
cin>>n>>k;
cout<<bfs()<<endl;//输出最小步数
return 0;
}