[LUOGU]1141 01迷宫
题目描述
有一个仅由数字0与1组成的n×n格迷宫。若你位于一格0上,那么你可以移动到相邻4格中的某一格1上,同样若你位于一格1上,那么你可以移动到相邻4格中的某一格0上。
你的任务是:对于给定的迷宫,询问从某一格开始能移动到多少个格子(包含自身)。
输入输出格式
输入格式:
输入的第1行为两个正整数n,m。
下面n行,每行n个字符,字符只可能是0或者1,字符之间没有空格。
接下来m行,每行2个用空格分隔的正整数i,j,对应了迷宫中第i行第j列的一个格子,询问从这一格开始能移动到多少格。
输出格式:
输出包括m行,对于每个询问输出相应答案。
输入输出样例
输入样例#1: 复制
2 2
01
10
1 1
2 2
输出样例#1: 复制
4
4
说明
所有格子互相可达。
对于20%的数据,n≤10;
对于40%的数据,n≤50;
对于50%的数据,m≤5;
对于60%的数据,n≤100,m≤100;
对于100%的数据,n≤1000,m≤100000。非常艰难.. 
第一次代码 暴力 70分
#include<iostream>
#include<string>
#include<cstring>
#define MAXN 1005
using namespace std;
int n,m;
int a[MAXN][MAXN];
int b[MAXN][MAXN];
bool vis[MAXN][MAXN];
int c[MAXN][MAXN];
int cnt;
int dx[4]={0,1,0,-1};
int dy[4]={-1,0,1,0};
void dfs(int x,int y){
    if(vis[x][y]) return;
    if(x<1||x>n||y<1||y>n) return;
    vis[x][y]=1;
    b[x][y]=1;
    cnt++;
    for(int i=0;i<=3;i++){
        int nx=x+dx[i];
        int ny=y+dy[i];
        if(a[nx][ny]==a[x][y]) continue;
        dfs(nx,ny);
    }
}
int calc(){
    int s=0;
    int i,j;
    for(i=1;i<=n;i++){
        for(j=1;j<=n;j++){
            if(b[i][j]) s++;
        }
    }
    return s;
}
void make(int num){
        for(int i=1;i<=n;i++){
        for(int j=1;j<=n;j++){
            if(b[i][j]) c[i][j]=num;
        }
    }
}
int main(){
    string s;
    cin>>n>>m;
    int i,j;
    for(i=1;i<=n;i++){
        cin>>s;
        for(j=0;j<n;j++){
            a[i][j+1]=s[j]-'0';
        }
    }
    int x,y;
    while(m--){
        cin>>x>>y;
        if(!vis[x][y]){
            dfs(x,y);
            cnt=calc();
            make(cnt);
            memset(b,0,sizeof(b));
        }
        cout<<c[x][y]<<endl;
    }
    return 0;
}cin改成scanf,试图写过getchar,不成功 80
#include<iostream>
#include<string>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#define MAXN 1005
using namespace std;
int n,m;
int a[MAXN][MAXN];
int b[MAXN][MAXN];
bool vis[MAXN][MAXN];
int c[MAXN][MAXN];
int cnt;
int dx[4]={0,1,0,-1};
int dy[4]={-1,0,1,0};
void dfs(int x,int y){
    if(vis[x][y]) return;
    if(x<1||x>n||y<1||y>n) return;
    vis[x][y]=1;
    b[x][y]=1;
    cnt++;
    for(int i=0;i<=3;i++){
        int nx=x+dx[i];
        int ny=y+dy[i];
        if(a[nx][ny]==a[x][y]) continue;
        dfs(nx,ny);
    }
}
int calc(){
    int s=0;
    for(register int i=1;i<=n;i++){
        for(register int j=1;j<=n;j++){
            if(b[i][j]) s++;
        }
    }
    return s;
}
void make(int num){
    for(register int i=1;i<=n;i++){
        for(register int j=1;j<=n;j++){
            if(b[i][j]) c[i][j]=num;
        }
    }
}
int main(){
    char s[1005];
    scanf("%d%d",&n,&m);
    register int i,j;
    for(i=1;i<=n;i++){
        scanf("%s",s);
        for(j=0;j<n;j++){
            a[i][j+1]=s[j]-'0';
        }
    }
//  for(int i=1;i<=n;i++){
//      for(int j=1;j<=n;j++){
//          a[i][j]=getchar()-'0';
//          if(a[i][j]<0||a[i][j]>1) j--;
//
//      }
//  }
    int x,y;
    while(m--){
        cin>>x>>y;
        if(!vis[x][y]){
            dfs(x,y);
            cnt=calc();
            make(cnt);
            memset(b,0,sizeof(b));
        }
        printf("%d\n",c[x][y]);
    }
    return 0;
}拍脑子一想,cnt在dfs时候就能记录了,没必要再来个n^2的循环。
#include<iostream>
#include<string>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#define MAXN 1005
using namespace std;
int n,m;
int a[MAXN][MAXN];
int b[MAXN][MAXN];
bool vis[MAXN][MAXN];
int c[MAXN][MAXN];
int cnt;
int dx[4]={0,1,0,-1};
int dy[4]={-1,0,1,0};
void dfs(int x,int y){
    if(vis[x][y]) return;
    if(x<1||x>n||y<1||y>n) return;
    vis[x][y]=1;
    b[x][y]=1;
    cnt++;
    for(register int i=0;i<=3;i++){
        int nx=x+dx[i];
        int ny=y+dy[i];
        if(a[nx][ny]==a[x][y]) continue;
        dfs(nx,ny);
    }
}
int calc(){
    int s=0;
    for(register int i=1;i<=n;i++){
        for(register int j=1;j<=n;j++){
            if(b[i][j]) s++;
        }
    }
    return s;
}
inline void make(int num){
    for(register int i=1;i<=n;i++){
        for(register int j=1;j<=n;j++){
            if(b[i][j]) c[i][j]=num;
        }
    }
}
int main(){
    char s[1005];
    scanf("%d%d",&n,&m);
    register int i,j;
    for(i=1;i<=n;i++){
        scanf("%s",s);
        for(j=0;j<n;j++){
            a[i][j+1]=s[j]-'0';
        }
    }
//  for(int i=1;i<=n;i++){
//      for(int j=1;j<=n;j++){
//          a[i][j]=getchar()-'0';
//          if(a[i][j]<0||a[i][j]>1) j--;
//
//      }
//  }
    int x,y;
    while(m--){
        cin>>x>>y;
        if(!vis[x][y]||!c[x][y]){
            cnt=0;
            dfs(x,y);
        //  cnt=calc();
            make(cnt);
            memset(b,0,sizeof(b));
        }
        printf("%d\n",c[x][y]);
    }
    return 0;
}然后想到没必要真的把每个点的值存进二位数组,因为无法同时和dfs完成,必须两步,慢。
但是可以存一个数字进去,在一次dfs中,这个数字代表的是一类点,再用ans数组建立数字和答案(cnt)的映射,但是ans开小了,又RE了一个点。。
AC代码:
#include<iostream>
#include<string>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#define MAXN 1005
using namespace std;
int n,m;
int a[MAXN][MAXN];
int b[MAXN][MAXN];
bool vis[MAXN][MAXN];
int c[MAXN][MAXN];
int cnt;
int dx[4]={0,1,0,-1};
int dy[4]={-1,0,1,0};
int ans[100005];
int p=1;
void dfs(int x,int y){
    if(vis[x][y]) return;
    if(x<1||x>n||y<1||y>n) return;
    vis[x][y]=1;
    c[x][y]=p;
    cnt++;
    for(register int i=0;i<=3;i++){
        int nx=x+dx[i];
        int ny=y+dy[i];
        if(a[nx][ny]==a[x][y]) continue;
        dfs(nx,ny);
    }
}
int main(){
    char s[1005];
    scanf("%d%d",&n,&m);
    register int i,j;
    for(i=1;i<=n;i++){
        scanf("%s",s);
        for(j=0;j<n;j++){
            a[i][j+1]=s[j]-'0';
        }
    }
    int x,y;
    while(m--){
        cin>>x>>y;
        if(!vis[x][y]){
            cnt=0;
            dfs(x,y);
            ans[p]=cnt;
            p++;
        }
        printf("%d\n",ans[c[x][y]]);
    }
    return 0;
}[LUOGU]1141 01迷宫的更多相关文章
- luogu P1141 01迷宫 x
		P1141 01迷宫 题目描述 有一个仅由数字0与1组成的n×n格迷宫.若你位于一格0上,那么你可以移动到相邻4格中的某一格1上,同样若你位于一格1上,那么你可以移动到相邻4格中的某一格0上. 你的任 ... 
- luogu P1141 01迷宫
		https://www.luogu.org/problem/show?pid=1141 还不太会用 BFS 然后就跟着感觉走了一波 经历了很多错误 刚开始的读入 然后BFS的过程 最后T三个点 看到别 ... 
- Luogu  P1141 01迷宫【搜索/dfs】By cellur925
		题目传送门 我tm到现在还需要刷这种水搜索...我退役吧. 但就是搜索弱嘛 补一补嘛qwq 题目大意:给你一张地图与许多询问,每次询问求这个点所在联通块的点的个数. 所以这个题目的本质就是在求联通块. ... 
- [洛谷Luogu]P1141 01迷宫[联通块 并查集]
		题目链接 大致题意 相邻格子不同为连通,计算每个点所在的连通块大小. 想法 我采用了并查集的做法. 开一个辅助数组记录连通块大小,每次合并的时候更新父亲节点的大小即可. 一个点先与它上面的点判定,若判 ... 
- [LuoguP1141]01迷宫
		1141 01迷宫 题目描述 有一个仅由数字0与1组成的n×n格迷宫.若你位于一格0上,那么你可以移动到相邻4格中的某一格1上,同样若你位于一格1上,那么你可以移动到相邻4格中的某一格0上. 你的任务 ... 
- 01迷宫 洛谷 p1141
		题目描述 有一个仅由数字0与1组成的n×n格迷宫.若你位于一格0上,那么你可以移动到相邻4格中的某一格1上,同样若你位于一格1上,那么你可以移动到相邻4格中的某一格0上. 你的任务是:对于给定的迷宫, ... 
- 洛谷——P1141 01迷宫
		P1141 01迷宫 题目描述 有一个仅由数字0与1组成的n×n格迷宫.若你位于一格0上,那么你可以移动到相邻4格中的某一格1上,同样若你位于一格1上,那么你可以移动到相邻4格中的某一格0上. 你的任 ... 
- 【u115】&&【t031】 01迷宫
		01迷宫(maze01) Time Limit: 1 second Memory Limit: 128 MB [问题描述] 有一个仅由数字0与1组成的n×n格迷宫.若你位于一格0上,那么你可以移动到相 ... 
- Codevs 1629 01迷宫
		1629 01迷宫 时间限制: 1 s 空间限制: 256000 KB 题目等级 : 钻石 Diamond 题目描述 Description 有一个由01组成的n*n格迷宫,若你位于一格0上,那么你可 ... 
随机推荐
- plt
			设定X,Y轴的长度以及刻度的方法. import numpy as np import matplotlib.pyplot as plt data = np.arange(0,1.1,0.01) pl ... 
- Java - Class版本号和UnsupportedClassVersionError
			问题分析 Java是向下兼容的,每一个jdk版本都有对应的class版本号(major + minor version numbers):如果用低版本的jvm去加载高版本jdk编译的类,就会报错:ja ... 
- SpringBoot | Velocity template
			SpringBoot版本: <parent> <groupId>org.springframework.boot</groupId> <artifactId& ... 
- 《http和https协议》
			一.HTTP协议 1.官方概念: HTTP协议是Hyper Text Transfer Protocol(超文本传输协议)的缩写,是用于从万维网(WWW:World Wide Web )服务器传输超文 ... 
- python 基础(十) 面向对象
			面向对象 一.概念 类(class): 用来描述具有相同属性和方法的对象的集合 对象是类的实例化 类变量:类变量在整个实例化的对象中是共用的.定义在类中 并且是函数体外的 实例变量:只能作用于 当前类 ... 
- B.选点
			链接:https://ac.nowcoder.com/acm/contest/368/B 题意: 有一棵n个节点的二叉树,1为根节点,每个节点有一个值wi.现在要选出尽量多的点. 对于任意一棵子树,都 ... 
- python之 __getattr__、__getattr__、__getitem__、__setitem__ 使用
			python之 __getattr__.__getattr__.__getitem__.__setitem__ 使用 __getattr__内置使用点号获取实例属性属性如 s.name,自调用__ge ... 
- Spark无法创建新线程
			Spark提交程序报错,无法创建新的线程 原因是因为这台公用机器上跑的进程太多了,需要修改Linux参数,允许用户最大进程数 查看允许用户最大进程数配置 ulimit -a 修改允许用户最大进程数配置 ... 
- Rabbitmq~linux环境的部署
			之前写过在windows环境上部署rabbitmq,这回介绍在centos上对这个消息中间件进行部署的过程 一 下载和解压 wget http://www.rabbitmq.com/releases ... 
- 遍历list集合的三种方式
			List<String> list1 = new ArrayList<String>(); list1.add("1"); list1.add(" ... 
