三.搜索与图论(未完结)

news2025/1/14 1:16:24

DFS(深搜)

之前写过三篇关于dfs的

练习总结:

基础算法--递归搜索DFS练习总结(上)-CSDN博客

基础算法--递归搜索DFS练习总结(中)-CSDN博客

基础算法--递归搜索DFS练习总结(下)-CSDN博客

以下题目均为

补充练习: 

P1460 [USACO2.1] 健康的荷斯坦奶牛 Healthy Holsteins 

第一种暴力搜索dfs(超时未AC 90 points):

#include<iostream>
#include<cstring>
using namespace std;
int n,m,a[30],b[20][30],c[30],ans;
bool flag[20],key;
bool check(){
    for(int i=1;i<=n;i++){
        for(int j=1;j<=m;j++) if(flag[j]) c[i]+=b[j][i];
        if(a[i]>c[i]) return false;
    }
    return true;
}
void dfs(int x,int cnt){
    if(x==cnt+1){
        if(check()) {key=true;ans=cnt;}
        memset(c,0,sizeof c);
        return ;
    }
    for(int i=x;i<=m;i++){
        if(!flag[i]){
            flag[i]=true;
            dfs(x+1,cnt);
            if(key) return ;
            flag[i]=false;
        }
    }
}
int main(){
    cin>>n;
    for(int i=1;i<=n;i++) cin>>a[i];
    cin>>m;
    for(int i=1;i<=m;i++) for(int j=1;j<=n;j++) cin>>b[i][j];
    for(int i=1;i<=m;i++){
        memset(flag,0,sizeof flag);
        memset(c,0,sizeof c);
        dfs(1,i);
        if(key){
            cout<<ans<<" ";
            for(int i=1;i<=m;i++) if(flag[i]) cout<<i<<" ";
            return 0;
        }
    }
}

第二种暴力搜索dfs(AC):

#include<iostream>
using namespace std;
int n,m,a[30],b[20][30],c[30],ans=1e9,res[30];
//数组c:每次搜索选的饲料编号,数组res:存储解
//判断每次选的那些饲料中的维生素之和是不是都大于等于牛需要的量 
bool check(int x){
    for(int i=1;i<=n;i++){
        int sum=0;
        for(int j=1;j<=x;j++) sum+=b[c[j]][i];
        if(sum<a[i]) return false;
    }
    return true;
}
void dfs(int curPos,int curCnt){
    if(curPos>m){//当前位置超过边界
        if(check(curCnt)){//必须使牛够吃
            if(curCnt<ans){//如果小于以前的最优解
                ans=curCnt;//更新答案
                for(int i=1;i<=m;i++) res[i]=c[i];//更新答案数组
            }
        }
        return ;
    }
    c[curCnt+1]=curPos;//把当前位置放在数组里
    dfs(curPos+1,curCnt+1);//搜索一步
    c[curCnt+1]=0;//回溯一步
    dfs(curPos+1,curCnt);//如果不选第curPos种饲料
}
int main(){
    cin>>n;
    for(int i=1;i<=n;i++) cin>>a[i];
    cin>>m;
    for(int i=1;i<=m;i++) for(int j=1;j<=n;j++) cin>>b[i][j];
    dfs(1,0);
    cout<<ans<<" ";
    for(int i=1;i<=ans;i++) cout<<res[i]<<" ";
    return 0;
}

P1451 求细胞数量

dfs(AC):

思路:不为0的地方就是细胞,然后dfs搜连通块,把搜到的都归0,保证不重复

#include<iostream>
using namespace std;
int n,m,ans;
char a[105][105];
int dx[4]={-1,0,1,0};
int dy[4]={0,1,0,-1};
void dfs(int x,int y){
    if(x<0||y<0||x>=n||y>=m) return ;
    a[x][y]='0';
    for(int i=0;i<4;i++){
        int xx=x+dx[i],yy=y+dy[i];
        if(xx>=0&&xx<n&&yy>=0&&yy<m){
            if(a[xx][yy]!='0'){
                dfs(xx,yy);
            }
        }
    }
}
int main(){
    cin>>n>>m;
    for(int i=0;i<n;i++) cin>>a[i];
    for(int i=0;i<n;i++) {
        for(int j=0;j<m;j++) {
            if(a[i][j]!='0'){
                dfs(i,j);
                ans++;
            }
        }
    }
    cout<<ans<<endl;
    return 0;
}

P1331 海战

dfs(AC):

和细胞一样的思路,唯一需要注意的是对于斜着连着的情况要特殊输出

#include<iostream>
using namespace std;
int n,m,ans;
char a[1005][1005];
int dx[4]={-1,0,1,0};
int dy[4]={0,1,0,-1};
bool flag;
void dfs(int x,int y){
    if(flag) return ;
    if(x<0||y<0||x>=n||y>=m) return ;
    a[x][y]='.';
    if(a[x-1][y+1]=='#'&&a[x-1][y]=='.'&&a[x][y+1]=='.') {flag=true; return ;}
    if(a[x+1][y+1]=='#'&&a[x+1][y]=='.'&&a[x][y+1]=='.') {flag=true; return ;}
    if(a[x+1][y-1]=='#'&&a[x+1][y]=='.'&&a[x][y-1]=='.') {flag=true; return ;}
    if(a[x-1][y-1]=='#'&&a[x-1][y]=='.'&&a[x][y-1]=='.') {flag=true; return ;}
    for(int i=0;i<4;i++){
        int xx=x+dx[i],yy=y+dy[i];
        if(xx>=0&&xx<n&&yy>=0&&yy<m){
            if(a[xx][yy]!='.'){
                dfs(xx,yy);
            }
        }
    }
}
int main(){
    cin>>n>>m;
    for(int i=0;i<n;i++) cin>>a[i];
    for(int i=0;i<n;i++) {
        for(int j=0;j<m;j++) {
            if(a[i][j]!='.'){
                dfs(i,j);
                ans++;
            }
        }
    }
    if(!flag) cout<<"There are "<<ans<<" ships."<<endl;
    else cout<<"Bad placement."<<endl;
    return 0;
}

B3625 迷宫寻路

 dfs(AC):

#include<iostream>
using namespace std;
int n,m;
char a[105][105];
int dx[4]={-1,0,1,0};
int dy[4]={0,1,0,-1};
void dfs(int x,int y){
    if(a[x-1][y]=='#'){
        if(a[x][y+1]=='#'){
            if(a[x+1][y]=='#'){
                if(a[x-1][y]=='#'){
                    return ;
                }
            }
        }
    }
    for(int i=0;i<4;i++){
        int xx=x+dx[i],yy=y+dy[i];
        if(xx>0&&xx<=n&&yy>0&&yy<=m&&a[xx][yy]=='.'){
            a[xx][yy]='#';
            dfs(xx,yy);
        }
    }
}
int main(){
    cin>>n>>m;
    for(int i=0;i<=m+1;i++) a[0][i]='#',a[n+1][i]='#';
    for(int i=1;i<=n;i++) a[i][0]='#',a[i][m+1]='#';
    for(int i=1;i<=n;i++) for(int j=1;j<=m;j++) cin>>a[i][j];
    dfs(1,1);
    if(a[n][m]=='#') cout<<"Yes"<<endl;
    else cout<<"No"<<endl;
    return 0;
}

 bfs(AC):

#include<iostream>
#include<queue>
#include<cstring>
using namespace std;
typedef pair<int,int> pii;
int n,m,d[105][105];
char a[105][105];
int dx[4]={-1,0,1,0};
int dy[4]={0,1,0,-1};
int bfs(){
    queue<pii>q;
    q.push({0,0});
    memset(d,-1,sizeof d);
    d[0][0]=0;
    while(q.size()){
        auto t=q.front();
        q.pop();
        for(int i=0;i<4;i++){
            int x=t.first+dx[i],y=t.second+dy[i];
            if(x>=0&&x<n&&y>=0&&y<m&&a[x][y]=='.'&&d[x][y]==-1){
                d[x][y]=d[t.first][t.second]+1;
                q.push({x,y});
            }
        }
    }
    return d[n-1][m-1];
}
int main(){
    cin>>n>>m;
    for(int i=0;i<n;i++) cin>>a[i];
    if(bfs()==-1) cout<<"No"<<endl;
    else cout<<"Yes"<<endl;
    return 0;
}

P1162 填涂颜色

思路:

利用dfs把原来的0变成1,并且标记数组记住原来的0,方便后续输出

然后图中剩下的1就是1,剩下的0就是2,输出即可

#include<iostream>
using namespace std;
const int N=35;
int n,a[N][N],flag[N][N];
int dx[4]={-1,0,1,0};
int dy[4]={0,1,0,-1};
void dfs(int x,int y){
    a[x][y]=1;
    flag[x][y]=1;
    if(a[x-1][y]==1){
        if(a[x][y+1]==1){
            if(a[x+1][y]==1){
                if(a[x][y-1]==1){
                    return ;
                }
            }
        }
    }
    for(int i=0;i<4;i++){
        int xx=x+dx[i],yy=y+dy[i];
        if(xx>0&&xx<=n&&yy>0&&yy<=n&&a[xx][yy]==0){
            dfs(xx,yy);
        }
    }
}
int main(){
    cin>>n;
    for(int i=0;i<=n+1;i++) a[0][i]=1,a[n+1][i]=1;
    for(int i=1;i<=n;i++) a[i][0]=-1,a[i][n+1]=-1;
    for(int i=1;i<=n;i++) for(int j=1;j<=n;j++) cin>>a[i][j];
    for(int i=1;i<=n;i++){
        if(a[1][i]==0) {flag[1][i]=1;dfs(1,i);}
        if(a[n][i]==0) {flag[n][i]=1;dfs(n,i);}
    }
    for(int i=1;i<=n-1;i++){
        if(a[i][1]==0) {flag[i][1]=1;dfs(i,1);}
        if(a[i][n]==0) {flag[i][n]=1;dfs(i,n);}
    }
    for(int i=1;i<=n;i++){
        for(int j=1;j<=n;j++){
            if(flag[i][j]) cout<<"0"<<" ";
            else{
                if(a[i][j]) cout<<"1"<<" ";
                else cout<<"2"<<" ";
            }
        }
        cout<<endl;
    }
    return 0;
}

P1506 拯救oibh总部

 和上面几题一样的思路dfs(AC):

#include<iostream>
using namespace std;
const int N=505;
int n,m,ans;
char a[N][N];
int dx[4]={-1,0,1,0};
int dy[4]={0,1,0,-1};
void dfs(int x,int y){
    a[x][y]='*';
    if(a[x-1][y]=='*'&&x-1>=0&&x-1<n&&y>=0&&y<m-1){
        if(a[x][y+1]=='*'&&x>=0&&x<n&&y+1>=0&&y+1<m-1){
            if(a[x+1][y]=='*'&&x+1>=0&&x+1<n&&y>=0&&y<m-1){
                if(a[x][y-1]=='*'&&x>=0&&x<n&&y-1>=0&&y-1<m-1){
                    return ;
                }
            }
        }
    }
    for(int i=0;i<4;i++){
        int xx=x+dx[i],yy=y+dy[i];
        if(a[xx][yy]=='0'&&xx>=0&&xx<n&&yy>=0&&yy<m){
            dfs(xx,yy);
        }
    }
}
int main(){
    cin>>n>>m;
    for(int i=0;i<n;i++) cin>>a[i];
    for(int i=0;i<m;i++){
        if(a[0][i]=='0') dfs(0,i);
        if(a[n-1][i]=='0') dfs(n-1,i);
    }
    for(int i=1;i<n-1;i++){
        if(a[i][0]=='0') dfs(i,0);
        if(a[i][m-1]=='0') dfs(i,m-1);
    }
    for(int i=0;i<n;i++) for(int j=0;j<m;j++) if(a[i][j]=='0') ans++;
    cout<<ans<<endl;
    return 0;
}

P8662 [蓝桥杯 2018 省 AB] 全球变暖

这题非常重要的一点是:原来的岛屿经过海水淹没后(只要没完全被淹没),仍然认为他们是同一座岛屿!!!

模拟下图样例:

海平面上升后:

答案为:4(原来有8个岛屿,红色是没有被完全淹没的,那么被完全淹没的就有4个)

因此思路是:先dfs所有原来的岛屿,接着dfs没被完全淹没的岛屿,最后相减得结果(AC):

#include<iostream>
using namespace std;
const int N=1005;
char a[N][N],b[N][N];
int n,res,ans;
int dx[4]={-1,0,1,0};
int dy[4]={0,1,0,-1};
void dfs_all(int x,int y){
    b[x][y]='.';
    if(b[x-1][y]=='.'&&x-1>=0&&x-1<n&&y>=0&&y<n){
        if(b[x][y+1]=='.'&&x>=0&&x<n&&y+1>=0&&y+1<n){
            if(b[x+1][y]=='.'&&x+1>=0&&x+1<n&&y>=0&&y<n){
                if(b[x][y-1]=='.'&&x>=0&&x<n&&y-1>=0&&y-1<n){
                    return ;
                }
            }
        }
    }
    for(int i=0;i<4;i++){
        int xx=x+dx[i],yy=y+dy[i];
        if(b[xx][yy]=='#'){
            dfs_all(xx,yy);
        }
    }
}
void dfs_rest(int x,int y){
    a[x][y]='.';
    if(a[x-1][y]=='.'&&x-1>=0&&x-1<n&&y>=0&&y<n){
        if(a[x][y+1]=='.'&&x>=0&&x<n&&y+1>=0&&y+1<n){
            if(a[x+1][y]=='.'&&x+1>=0&&x+1<n&&y>=0&&y<n){
                if(a[x][y-1]=='.'&&x>=0&&x<n&&y-1>=0&&y-1<n){
                    return ;
                }
            }
        }
    }
    for(int i=0;i<4;i++){
        int xx=x+dx[i],yy=y+dy[i];
        if(a[xx][yy]=='*'||a[xx][yy]=='#'){
            dfs_rest(xx,yy);
        }
    }
}
int main(){
    cin>>n;
    for(int i=0;i<n;i++) for(int j=0;j<n;j++) {cin>>a[i][j];b[i][j]=a[i][j];}
    for(int i=0;i<n;i++) for(int j=0;j<n;j++) if(b[i][j]=='#') {dfs_all(i,j);res++;}
    for(int i=0;i<n;i++){
        for(int j=0;j<n;j++){
            if(a[i][j]=='.'){
                for(int k=0;k<4;k++){
                    int x=i+dx[k],y=j+dy[k];
                    if(a[x][y]=='#') a[x][y]='*';
                }
            }
        }
    }
    for(int i=0;i<n;i++) for(int j=0;j<n;j++) if(a[i][j]=='#') {dfs_rest(i,j);ans++;}
    cout<<res-ans<<endl;
    return 0;
}

BFS(广搜)

模拟队列模板: 

typedef pair<int,int>pii;
int mapp[N][N];
int d[N][N];//每一个点到起点的距离
pii q[N*N];//手写队列
int bfs(){
    int hh=0,tt=0;
    q[0]={0,0};
    memset(d,-1,sizeof d);//距离初始化为-1表示没有走过
    d[0][0]=0;//表示起点走过了
    int dx[4]={-1,0,1,0},dy[4]={0,1,0,-1};//方向向量
    while(hh <= tt){//队列不空
        auto t=q[hh++];//取走队头元素
        for(int i=0;i<4;i++){//枚举4个方向
            int x=t.first+dx[i],y=t.second+dy[i];//更新方向
            if(x>=0&&x<n&&y>=0&&y<m&&mapp[x][y]==0&&d[x][y]==-1){//在边界内并且是空地并且之前没有走过
                d[x][y]=d[t.first][t.second]+1;//更新距离
                q[++tt]={x,y};//新坐标入队
            }
        }
    }
    return d[n-1][m-1];//右下角距起点的距离
}

STL模板:

typedef pair<int,int> pii;
int mapp[N][N],d[N][N];
int bfs(){
    queue<pii>q;
    q.push({0,0});
    memset(d,-1,sizeof d);
    d[0][0]=0;
    int dx[4]={-1,0,1,0},dy[4]={0,1,0,-1};
    while(q.size()){
        auto t=q.front();
        q.pop();
        for(int i=0;i<4;i++){
            int x=t.first+dx[i],y=t.second+dy[i];
            if(x>=0&&x<n&&y>=0&&y<m&&mapp[x][y]==0&&d[x][y]==-1){
                d[x][y]=d[t.first][t.second]+1;
                q.push({x,y});
            }
        }
    }
    return d[n-1][m-1];
}

打印路径模板:

typedef pair<int,int> pii;
int mapp[N][N],d[N][N];
pii preNode[N][N];//存储每个点的前一个点的坐标,方便追踪路径
int bfs(){
    queue<pii>q;
    q.push({0,0});
    memset(d,-1,sizeof d);
    d[0][0]=0;
    int dx[4]={-1,0,1,0},dy[4]={0,1,0,-1};
    while(q.size()){
        auto t=q.front();
        q.pop();
        for(int i=0;i<4;i++){
            int x=t.first+dx[i],y=t.second+dy[i];
            if(x>=0&&x<n&&y>=0&&y<m&&mapp[x][y]==0&&d[x][y]==-1){
                d[x][y]=d[t.first][t.second]+1;
                preNode[x][y]=t;//记录前驱点,是由t走到(x,y)的
                q.push({x,y});
            }
        }
    }
    return d[n-1][m-1];
}
void print_path(){
    if(d[n-1][m-1]==-1){
        cout<<"无法抵达终点"<<endl;
        return ;
    }
    vector<pii>path;
    //从终点遍历追踪
    for(pii x={n-1,m-1};x!=make_pair(0,0);x=preNode[x.first][x.second]) path.push_back(x);
    path.push_back({0,0});//加入起点
    reverse(path.begin(), path.end());//将路径逆序
    for(auto x:path) cout<<"("<<x.first<<","<<x.second<<")"<<"->";
    cout<<"终点"<<endl;
}

bfs练习:

B3626 跳跃机器人

一维路径问题 bfs(AC):

#include<iostream>
#include<queue>
#include<cstring>
using namespace std;
const int N=1e6+5;
int n,d[N];
int bfs(){
    queue<int>q;
    q.push({1});
    memset(d,-1,sizeof d);
    d[1]=0;
    while(q.size()){
        auto t=q.front();
        q.pop();
        if(t-1>=1&&t-1<=n&&d[t-1]==-1){
            d[t-1]=d[t]+1;
            q.push({t-1});
        }
        if(t+1>=1&&t+1<=n&&d[t+1]==-1){
            d[t+1]=d[t]+1;
            q.push({t+1});
        }
        if(2*t>=1&&2*t<=n&&d[2*t]==-1){
            d[2*t]=d[t]+1;
            q.push({2*t});
        }
    }
    return d[n];
}
int main(){
    cin>>n;
    cout<<bfs()<<endl;
    return 0;
}

P2802 回家

二维路径问题+多种情况的判断 bfs(AC):

#include<iostream>
#include<cstring>
#include<queue>
using namespace std;
const int N=10;
int n,m,xbegin,ybegin;
struct point{   //结构体存储当前坐标状态
    int px,py,pstep,php;//坐标,步数,血量
};
int a[N][N];    //地图
bool flag[N][N];//标记经过鼠标
int dx[4]={-1,0,1,0};
int dy[4]={0,1,0,-1};
int bfs(int x,int y){
    queue<point>q;         //注意类型是结构体
    q.push(point{x,y,0,6});//出发点入队
    while(q.size()){
        auto t=q.front();  //取队头元素
        int x=t.px,y=t.py,step=t.pstep,hp=t.php;//初始化各变量
        q.pop();           //出队
        if(a[x][y]==3) return step;//走到家就结束
        if(hp>1){          //hp<=1时直接死亡
            for(int i=0;i<4;i++){
                int xx=x+dx[i],yy=y+dy[i];
                if(xx>=0&&xx<n&&yy>=0&&yy<m){//不能出界
                    if(a[xx][yy]==1||a[xx][yy]==3){//路和家直接走
                        q.push(point{xx,yy,step+1,hp-1});//入队
                    }
                    if(a[xx][yy]==4&&!flag[xx][yy]){//有鼠标的点最多只能走一次
                        flag[xx][yy]=true;//标记
                        q.push(point{xx,yy,step+1,6});//入队
                    }
                }
            }
        }
    }
    return -1;//无解返回-1
}
int main(){
    cin>>n>>m;
    for(int i=0;i<n;i++){
        for(int j=0;j<m;j++){
            cin>>a[i][j];
            if(a[i][j]==2) xbegin=i,ybegin=j;
        }
    }
    cout<<bfs(xbegin,ybegin)<<endl;
    return 0;
}

P1135 奇怪的电梯

 这题会卡纯的dfs,当然也可以搜索每一条路后就更新答案再继续搜索(未AC 64points):

#include<iostream>
using namespace std;
int n,a,b,k[205],ans=1e6,flag[205];
//当前在第x楼,按了cnt次按钮
void dfs(int x,int cnt){
    if(cnt>=ans) return ;
    if(x==b){
        ans=min(ans,cnt);
        return ;
    }
    flag[x]=1;
    if(x+k[x]<=n&&flag[x+k[x]]==0){ //上楼
        flag[x+k[x]]=1;
        dfs(x+k[x],cnt+1);
        flag[x+k[x]]=0;
    }
    if(x-k[x]>=1&&flag[x-k[x]]==0){ //下楼
        flag[x-k[x]]=1;
        dfs(x-k[x],cnt+1);
        flag[x-k[x]]=0;
    }
}
int main(){
    cin>>n>>a>>b;
    for(int i=1;i<=n;i++) cin>>k[i];
    dfs(a,0);
    if(ans==1e6) {cout<<"-1"<<endl; return 0;}
    cout<<ans<<endl;
    return 0;
}

bfs(AC):

#include<iostream>
#include<queue>
using namespace std;
const int N=205;
int n,A,B,lift[N];
bool flag[N];
//标记数组作用是保证当前位置没有来过,因为如果回到原来经过的位置,那么就会一直重复
struct point{
    int loc,ans;//位置,当前已经走的步数
};
int bfs(int a,int b){
    queue<point>q;
    q.push(point{a,0});
    while(q.size()){
        auto t=q.front();
        int location=t.loc,step=t.ans;
        q.pop();
        if(location==b) return step;//到达终点
        if(location+lift[location]<=n){//可以向上
            if(!flag[location+lift[location]]){//并且该位置没有来过
                flag[location+lift[location]]=true;
                q.push(point{location+lift[location],step+1});
            }
        }
        if(location-lift[location]>=1){//可以向下
            if(!flag[location-lift[location]]){//并且该位置没有来过
                flag[location-lift[location]]=true;
                q.push(point{location-lift[location],step+1});
            }
        }
    }
    return -1;
}
int main(){
    cin>>n>>A>>B;
    for(int i=1;i<=n;i++) cin>>lift[i];
    cout<<bfs(A,B)<<endl;
    return 0;
}

P1747 好奇怪的游戏

这题注意初始化flag标记数组,并且把数组开大一点就没问题了

bfs(AC):

#include<iostream>
#include<cstring>
#include<queue>
using namespace std;
const int N=50;
int x1,y1,x2,y2;
int dx[12]={-2,-2,-1,-2,1,-2,2,-1,2,1,2,2};
int dy[12]={-2,-1,-2,1,-2,2,-2,2,-1,2,1,2};
bool flag[N][N];
struct point{
    int px,py,pstep;//坐标,步数
};
int bfs(int a,int b){
    queue<point>q;
    q.push(point{a,b,0});
    memset(flag,0,sizeof flag);
    flag[a][b]=true;
    while(q.size()){
        auto t=q.front();
        q.pop();
        int x=t.px,y=t.py,step=t.pstep;
        if(x==1&&y==1) return step;
        for(int i=0;i<12;i++){
            int xx=x+dx[i],yy=y+dy[i];
            if(xx>=1&&yy>=1){
                if(!flag[xx][yy]){
                    flag[xx][yy]=true;
                    q.push(point{xx,yy,step+1});
                }
            }
        }
    }
}
int main(){
    cin>>x1>>y1>>x2>>y2;
    cout<<bfs(x1,y1)<<endl;
    cout<<bfs(x2,y2)<<endl;
    return 0;
}

P1443 马的遍历

方法类似,只需要额外开一个二维答案数组存储步数即可

bfs(AC):

#include<iostream>
#include<cstring>
#include<queue>
using namespace std;
const int N=405;
int n,m,x,y;
int dx[8]={-2,-1,-2,1,-1,2,1,2};
int dy[8]={-1,-2,1,-2,2,-1,2,1};
int ans[N][N];//答案矩阵
bool flag[N][N];
struct point{
    int px,py;//坐标
};
void bfs(int a,int b){
    ans[a][b]=0;
    flag[a][b]=true;
    queue<point>q;
    q.push(point{a,b});
    while(q.size()){
        auto t=q.front();
        q.pop();
        int x=t.px,y=t.py;
        for(int i=0;i<8;i++){
            int xx=x+dx[i],yy=y+dy[i];
            if(xx>=1&&xx<=n&&yy>=1&&yy<=m){
                if(!flag[xx][yy]){
                    flag[xx][yy]=true;
                    q.push(point{xx,yy});
                    ans[xx][yy]=ans[x][y]+1;
                }
            }
        }
    }
    return ;
}
int main(){
    cin>>n>>m>>x>>y;
    memset(ans,-1,sizeof ans);
    memset(flag,0,sizeof flag);
    bfs(x,y);
    for(int i=1;i<=n;i++){
        for(int j=1;j<=m;j++){
            cout<<ans[i][j]<<" ";
        }
        cout<<endl;
    }
    return 0;
}

P1746 离开中山路

bfs(AC):

#include<iostream>
#include<cstring>
#include<queue>
using namespace std;
const int N=1005;
int n,x1,x2,y1,y2;
char a[N][N];
int dx[4]={-1,0,1,0};
int dy[4]={0,1,0,-1};
bool flag[N][N];
struct point{
    int px,py,ps;
};
int bfs(int bx,int by,int ex,int ey){
    queue<point>q;
    q.push(point{bx,by,0});
    flag[bx][by]=true;
    while(q.size()){
        auto t=q.front();
        int x=t.px,y=t.py,step=t.ps;
        q.pop();
        if(x==ex&&y==ey) return step;
        for(int i=0;i<4;i++){
            int xx=x+dx[i],yy=y+dy[i];
            if(xx>=1&&xx<=n&&yy>=1&&yy<=n){
                if(a[xx][yy]=='0'){
                    if(!flag[xx][yy]){
                        flag[xx][yy]=true;
                        q.push(point{xx,yy,step+1});
                    }
                }
            }
        }
    }
    return -1;
}
int main(){
    cin>>n;
    for(int i=1;i<=n;i++) for(int j=1;j<=n;j++) cin>>a[i][j];
    cin>>x1>>y1>>x2>>y2;
    cout<<bfs(x1,y1,x2,y2)<<endl;
    return 0;
}

P3395 路障

这题整体思路不变,唯一要加的是唯一需要注意的是路障的处理,可开一个block数组存放在(x,y)位置路障将在什么时候放下,对于没有路障的点,直接走过,对于有路障的点,判断时间即可

bfs(AC):

#include<iostream>
#include<cstring>
#include<queue>
using namespace std;
const int N=1005;
int T,n,a[N][N],block[N][N];
int dx[4]={-1,0,1,0};
int dy[4]={0,1,0,-1};
bool flag[N][N];
struct point{
    int px,py,pt;//坐标,时间
};
bool bfs(){
    queue<point>q;
    q.push(point{1,1,0});
    flag[1][1]=true;
    while(!q.empty()){
        auto t=q.front();
        int x=t.px,y=t.py,timee=t.pt;
        q.pop();
        if(x==n&&y==n) return true;
        timee++;
        for(int i=0;i<4;i++){
            int xx=x+dx[i],yy=y+dy[i];
            if(xx>=1&&xx<=n&&yy>=1&&yy<=n){
                if(a[xx][yy]==0&&!flag[xx][yy]){
                    flag[xx][yy]=true;
                    if(!block[xx][yy]||timee<=block[xx][yy]){
                        q.push(point{xx,yy,timee});
                    }
                }
            }
        }
    }
    return false;
}
int main(){
    cin>>T;
    while(T--){
        memset(flag,0,sizeof flag);
        memset(block,0,sizeof block);
        memset(a,0,sizeof a);
        cin>>n;
        for(int i=1;i<=2*n-2;i++){
            int a1,b1;
            cin>>a1>>b1;
            block[a1][b1]=i;
        }
        if(bfs()) cout<<"Yes"<<endl;
        else cout<<"No"<<endl;
    }
    return 0;
}

P8673 [蓝桥杯 2018 国 C] 迷宫与陷阱

和 P2802 回家 类似,

三维状态数组+多种情况判断 bfs(AC):

#include<iostream>
#include<cstring>
#include<queue> 
using namespace std;
const int N=1005;
int n,k;
char a[N][N];
int dx[4]={-1,0,1,0};
int dy[4]={0,1,0,-1};
//flag[x][y][z]:在剩余z个无敌时间时,有没有经过(x,y)
bool flag[N][N][11];
struct point{
    int px,py,ps,pk;//坐标,当前走的步数,剩余无敌的步数
};
int bfs(){
    queue<point>q;
    q.push({1,1,0,0});
    flag[1][1][0]=true;
    while(!q.empty()){
        auto t=q.front();
        q.pop();
        int x=t.px,y=t.py,step=t.ps,temp=t.pk;
        if(x==n&&y==n) return step;//到达终点,直接结束
        for(int i=0;i<4;i++){
            int xx=x+dx[i],yy=y+dy[i];
            if(xx>=1&&xx<=n&&yy>=1&&yy<=n){//保证边界
                if(a[xx][yy]!='#'){//墙是永远不可能经过的
                    //当前是未使用过的道具
                    if(a[xx][yy]=='%'&&!flag[xx][yy][k]){
                        flag[xx][yy][k]=true;
                        q.push({xx,yy,step+1,k});
                    }
                    //当前状态是无敌,道路和陷阱都可以走
                    if(temp>0&&!flag[xx][yy][temp-1]){
                        flag[xx][yy][temp-1]=true;
                        q.push({xx,yy,step+1,temp-1});
                    }
                    //当前状态不是无敌,道路可以走
                    if(temp==0&&a[xx][yy]=='.'&&!flag[xx][yy][0]){
                        flag[xx][yy][0]=true;
                        q.push({xx,yy,step+1,0});
                    }
                }
            }
        }
    }
    return -1;
}
int main(){
    cin>>n>>k;
    for(int i=1;i<=n;i++) for(int j=1;j<=n;j++) cin>>a[i][j];
    cout<<bfs()<<endl;
    return 0;
}

P1141 01迷宫

暴力bfs (超时未AC 70points):

#include<iostream>
#include<cstring>
#include<queue>
using namespace std;
const int N=1e3+5;
char a[N][N];
bool flag[N][N];
int n,m,ans;
int dx[4]={-1,0,1,0};
int dy[4]={0,-1,0,1};
int bfs(int tx,int ty){
    ans=1;
    flag[tx][ty]=1;
    queue<pair<int,int>>q;
    q.push({tx,ty});
    while(!q.empty()){
        auto t=q.front();
        int x=t.first,y=t.second;
        q.pop();
        for(int i=0;i<4;i++){
            int xx=x+dx[i],yy=y+dy[i];
            if(xx>=1&&xx<=n&&yy>=1&&yy<=n){
                if(!flag[xx][yy]&&a[xx][yy]!=a[x][y]){
                    flag[xx][yy]=1;
                    ans++;
                    q.push({xx,yy});
                }
            }
        }
    }
    return ans;
}
int main(){
    cin>>n>>m;
    for(int i=1;i<=n;i++) for(int j=1;j<=n;j++) cin>>a[i][j];
    while(m--){
        int ax,bx;
        cin>>ax>>bx;
        memset(flag,0,sizeof flag);
        cout<<bfs(ax,bx)<<endl;
    }
    return 0;
}

优化思路:

联通块上每一个点的答案等于该联通块的数目

注意这里的联通块指的是四个方向上0,1相间隔的联通块,如下图,在搜 a[1][2] 时会经过6个点,而这6个点的答案是一样的,都是6,因此通过搜完联通块直接给联通块上的所有点存答案即可

 

因此可以考虑 dfs+记忆化搜索(AC):

#include<iostream>
using namespace std;
const int N=1e3+5;
char a[N][N];//地图
int ans[N][N];//答案数组为0表示没有搜过,大于0表示答案
int temp[N*N][2];//记忆化,存储坐标数组
int n,m,res;
int dx[4]={-1,0,1,0};
int dy[4]={0,-1,0,1};
void dfs(int x,int y){
    res++;//搜到,答案自增
    //记忆化:存储坐标
    temp[res][0]=x;
    temp[res][1]=y;
    //ans数组当前只起标记作用
    ans[x][y]=1;
    for(int i=0;i<4;i++){
        int xx=x+dx[i],yy=y+dy[i];
        if(xx>=1&&xx<=n&&yy>=1&&yy<=n){
            if(a[x][y]!=a[xx][yy]){
                if(!ans[xx][yy]){
                    dfs(xx,yy);
                }
            }
        }
    }
    return ;
}
int main(){
    cin>>n>>m;
    for(int i=1;i<=n;i++) for(int j=1;j<=n;j++) cin>>a[i][j];
    while(m--){
        int ax,bx;
        cin>>ax>>bx;
        res=0;
        //已经存储过答案直接输出
        if(ans[ax][bx]) cout<<ans[ax][bx]<<endl;
        else{
            //否则搜素
            dfs(ax,bx);
            //记忆化:把路径上所有点的答案存储
            //ans数组正式记录所有答案
            for(int i=1;i<=res;i++)
                ans[temp[i][0]][temp[i][1]]=res;
            cout<<res<<endl;
        }
    }
    return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1656126.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【隧道篇 / WAN优化】(7.4) ❀ 03. WAN优化的原理 ❀ FortiGate 防火墙

【简介】相信对WAN优化感兴趣的人都会有疑问&#xff0c;WAN优化真的有作用吗&#xff1f;如果真的有作用&#xff0c;那是根据什么原理呢&#xff1f;让我们来更深入的了解一下。 客户端和服务器端 其实很多人在一开始看到WAN优化这个词&#xff0c;就自然的以为上网速度太慢&…

C++学习第十三天(多态)

1、多态的概念 概念 就是指多种形态&#xff0c;具体点就是去完成某个行为&#xff0c;当不同的对象去完成时会产生不同的状态 2、多态的定义及实现 多态的构成条件 多态是在不同继承关系的类对象&#xff0c;去调用同一函数&#xff0c;产生了不同的行为。 要在继承中构成…

linux权限维持-HackerPermKeeper

&#x1f512; HackerPermKeeper 通过渗透拿到权限之后&#xff0c;为了不让权限丢失&#xff0c;都会进行权限维持&#xff0c;而在进行权限维持的时候&#xff0c;红队需要花费大量的时候&#xff0c;来验证是否合适&#xff0c;因此在这款工具就诞生 HackerPermKeeper[黑客…

【动态规划】子数组、子串系列I|最大子数组和|环形子数组的最大和|乘积最大子数组|乘积为正数的最长子数组长度

一、最大子数组和 最大子数组和 算法原理&#xff1a; &#x1f4a1;细节&#xff1a; 1.返回值为dp表每个位置的最大值&#xff0c;而不是只看最后一个位置&#xff0c;因为可能最后一个位置都不选 2.可以直接在填dp表的时候就进行返回值的比较 3.如果初始化选择多开一个位…

vue+lodop实现web端打印标签功能

背景&#xff1a;项目要求在web端连接标签打印机&#xff0c;打印收件人信息 lodop打印插件地址&#xff1a;Lodop和C-Lodop官网主站 在项目中使用 1、去官网下载lodop包下载中心 - Lodop和C-Lodop官网主站 windows系统直接下载windows32版的就可以 2、解压安装 点击CLodop…

gtest的编译与使用

文章目录 gtest的编译与使用概述笔记CMake参数官方文档测试程序测试效果END gtest的编译与使用 概述 gTest是 googletest的缩写&#xff0c;如果直接找gTest项目&#xff0c;是找不到的。 库地址 https://github.com/google/googletest.git 迁出到本地后&#xff0c;切到最新…

vue中使用element的i18n语言转换(保姆式教程-保证能用)

1、项目中需要使用的插件&#xff0c;vue2或vue3、element、vue-i18n、js-cookie、vuex我是在vue2中使用 npm i element-ui -S npm i js-cookie -S npm i vue-i18n8.28.2 //因为我项目使用的vue2&#xff0c;直接安装报错了,就下载了固定的版本2、在main.js中引入i18n impor…

买手机参考

买手机一定要看cpu排行榜&#xff0c;避免受外界宣传或干扰等因素等的影响&#xff0c;导致买家消费者被割韭菜。是高等手机还是低等手机&#xff0c;cpu排行榜就是照妖镜。cpu综合处理能力&#xff08;兼顾功耗&#xff09;排行榜如图。 第二张 第三张 第四张

C#语言核心

一、面向对象基本概念 万物皆对象&#xff0c;用程序来抽象&#xff08;形容&#xff09;对象&#xff0c;用面向对象的思想来编程 用中文去形容一类对象&#xff0c;把一类对象的共同点提取出来&#xff0c;然后用程序语言把它翻译过来&#xff0c;带着对象的概念在程序中使…

Excel 查找值的位置后再用位置取值Excel处理

例题描述 Excel 文件中有下图所示的 3 个片区数据 (不同颜色标明)。 现在要算出1-12对应的一列数据&#xff0c;计算规则&#xff1a;在片区3中依次查找1-12&#xff0c;找到后在片区1对应位置取数&#xff0c;如果是0则取片区2同位置的数&#xff0c;如果是1则取F当前查找数。…

docker-compose管理jenkins

1.安装docker和compose 1.docker 更新系统&#xff1a;yum update 安装依赖项&#xff1a;yum install -y yum-utils device-mapper-persistent-data lvm2 配置镜像源&#xff1a;yum-config-manager --add-repo http://mirrors.aliyun.com/docker-ce/linux/centos/docker-ce…

出货300万片后,智舱界「小高通」浮出水面

‍作者 |张祥威 编辑 |德新 2024年北京车展&#xff0c;本土芯片公司开始截击外企供应商。 很长一段时间内&#xff0c;汽车行业智驾芯片看英伟达&#xff0c;座舱芯片看高通。英伟达Orin系列广受欢迎&#xff0c;高通8155席卷主流智能汽车&#xff0c;8295更是被视为最强配置…

Seata之AT 模式的使用

系列文章目录 文章目录 系列文章目录前言 前言 前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;忍不住分享一下给大家。点击跳转到网站&#xff0c;这篇文章男女通用&#xff0c;看懂了就去分享给你的码吧。 Seata 是一款开源的…

STM32F4xx开发学习_USART串口通讯

USART串口通讯 USART简介 USART&#xff08;universal synchronous asynchronous receiver transmitter&#xff09;&#xff0c;通用同步异步接收发射机&#xff0c;是一种全双工异步通信串行通讯方式&#xff0c;是STM32内部集成的硬件外设&#xff0c;以帧格式传输数据。搭…

NAT网络基本原理和认识,内网穿透的必备知识

NAT的基本介绍 NAT&#xff08;Network Address Translation&#xff09;是网络地址转换&#xff0c;它是一个IETF(Internet Engineering Task Force, Internet工程任务组)标准&#xff0c;允许一个整体机构以一个公用IP&#xff08;Internet Protocol&#xff09;地址出现在I…

二总线,替代传统485总线通讯,主站设计

二总线通信设计专栏 《二总线&#xff0c;替代传统485总线通讯&#xff0c;选型及应用-CSDN博客》《二总线&#xff0c;替代传统485总线通讯&#xff0c;低成本直流载波方案实现及原理-CSDN博客》《二总线&#xff0c;替代传统485总线通讯&#xff0c;调试避坑指南之最大的电流…

探索鸿蒙开发:鸿蒙系统如何引领嵌入式技术革新

嵌入式技术已经成为现代社会不可或缺的一部分。而在这个领域&#xff0c;华为凭借其自主研发的鸿蒙操作系统&#xff0c;正悄然引领着一场技术革新的浪潮。本文将探讨鸿蒙开发的特点、优势以及其对嵌入式技术发展的深远影响。 鸿蒙操作系统的特点 鸿蒙&#xff0c;作为华为推…

【Java】从0实现一个基于SpringBoot的个人博客系统

从0实现一个基于SpringBoot的个人博客系统 项目介绍准备工作数据准备创建项目准备前端页面编写配置文件 项目公共模块实体类公共层业务代码持久层实现博客列表实现博客列表约定前后端交互接口 实现博客详情约定前后端交互接口实现服务器代码 实现登录JWT令牌JWT令牌生成和校验实…

详解如何把文件或应用开机自启动

相信很多人都想把某些开机都要用的软件或文件打开&#xff0c;但不知道如何操作&#xff0c;或不知道可以手动设置开机自启动这一操作&#xff0c;下面为大家讲解实现思路&#xff1a; 1.首先打开winR打开运行框&#xff0c;在里面弄输入&#xff1a;shell:startup 2.回车进入文…

Win10/11共享文件夹,访问提示需要输入用户名密码

Win10/11共享文件夹&#xff0c;访问提示需要输入用户名密码 问题 已经关闭了密码保护共享&#xff0c;但是局域网其他电脑访问该文件夹&#xff0c;提示需要输入用户名和密码 解决方法 操作步骤 1.按WINR键打开运行&#xff0c;输入gpedit.msc打开本地组策略编辑器 2.按如…