[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上,那么你可 ...
随机推荐
- [SDOI2019] 热闹又尴尬的聚会
热闹度\(p\)子图中最小的度数,尴尬度\(q\)独立集大小,之间的约束 \[ \begin{aligned} \lfloor n/(p+1)\rfloor\le q &\rightarrow ...
- 新装ubuntu 12.04 , 使用技巧
*********************************************** 一.让Ubuntu 12.04开机默认进入命令行模式. 修改 /etc/default/grubGRUB ...
- sql数据库查询结果字段包含换行符导致复制到Excel发生错位问题的解决
问题描述:在工作过程中,有时会遇到这样的问题,写好sql查询语句在数据库中查询数据,看到行数(比如说是1000行),但是把查询结果复制到Excel里面,却发生了行列错位问题,而导致Excel里面的行数 ...
- 浅谈volatile关键字
volatile是一种轻量级的同步机制.它可以保证内存可见性以及防止指令重排序,但是不保证原子性 volatile和JMM机制是不可分割的,在谈volatile的时候有必要先了解以下JMM JMM(J ...
- redis连接错误
连接redis错误:ERR Client sent AUTH, but no password is set 2018-07-04 20:33 by robinli, 4367 阅读, 0 评论, 收 ...
- JDBC连接池一 自定义连接池
package com.mozq.jdbc; import java.io.IOException; import java.io.InputStream; import java.sql.Conne ...
- Git - .gitignore文件的用法
.gitignore文件的作用 .gitignore文件用来忽略被指定的文件或文件夹的改动,被记录在.gitignore文件里的文件或文件夹,是无法被git跟踪到的,换句话说,被忽略的文件是不会被放入 ...
- git合并某次提交到某个分支
有的时候,在develop分支开发,是大家公用的开发分支,但是只想合并自己提交的到master,如何操作呢?那就要用cherry-pick了. 语法 git cherry-pick commitid ...
- 描述符__get__,__set__,__delete__和析构方法__del__
描述符__get__,__set__,__delete__ 1.描述符是什么:描述符本质就是一个新式类,在这个新式类中,至少实现了__get__(),__set__(),__delete__()中的一 ...
- [BZOJ3128]a+b Problem
题解 最小割+主席树优化建图 首先看到每个点只有\(0/1\)两种状态就想到最小割 然后由于有一个限制是点\(i\)是黑点且有符合条件的白点就会减去\(p_i\) 所以我们将\(S\)集合设为黑点集合 ...