1.上午(2h)
将bfs看了很多遍,把涂色题码了一大半,但是还是差一点过
题目描述
由数字00组成的方阵中,有一任意形状闭合圈,闭合圈由数字11构成,围圈时只走上下左右44个方向。现要求把闭合圈内的所有空间都填写成22.例如:6 \times 66×6的方阵(n=6n=6),涂色前和涂色后的方阵如下:
0 0 0 0 0 0
0 0 1 1 1 1
0 1 1 0 0 1
1 1 0 0 0 1
1 0 0 0 0 1
1 1 1 1 1 1
0 0 0 0 0 0
0 0 1 1 1 1
0 1 1 2 2 1
1 1 2 2 2 1
1 2 2 2 2 1
1 1 1 1 1 1
输入格式
每组测试数据第一行一个整数n(1 \le n \le 30)n(1≤n≤30)
接下来nn行,由00和11组成的n \times nn×n的方阵。
方阵内只有一个闭合圈,圈内至少有一个00。
//感谢黄小U饮品指出本题数据和数据格式不一样. 已修改(输入格式)
输出格式
已经填好数字22的完整方阵。
输入输出样例
输入 #1复制
6 0 0 0 0 0 0 0 0 1 1 1 1 0 1 1 0 0 1 1 1 0 0 0 1 1 0 0 0 0 1 1 1 1 1 1 1
输出 #1复制
0 0 0 0 0 0 0 0 1 1 1 1 0 1 1 2 2 1 1 1 2 2 2 1 1 2 2 2 2 1 1 1 1 1 1 1
说明/提示
1 \le n \le 301≤n≤30
这道题和其他涂色题相比是有可以投机取巧的地方的
可以先把最外层加一圈0,然后将0分成两种,一种是在1圈外面的,另一种就是在1圈里面的,由于题目的表述限制,是可以过的
#include<stdio.h>
int n;
int map[35][35];//地图
int vis[35][35];//标记
int fx[4]={0,0,-1,1};//移动
int fy[4]={1,-1,0,0};
void dfs(int x, int y)
{
if(x<0||y<0||x>n+1||y>n+1||vis[x][y]!=0)//判断是否越界或者为墙
return ;
vis[x][y]=1;//标记
for(int i=0;i<=3;i++)
dfs(x+fx[i],y+fy[i]);//进一步搜
}
int main()
{
scanf("%d",&n);//输入
for(int i=1;i<=n;i++)
for(int j=1;j<=n;j++)
{
scanf("%d",&map[i][j]);//输入地图
if(map[i][j]==0)
vis[i][j]=0;
else
vis[i][j]=2;//方便打印
}
dfs(0,0);
for(int i=1;i<=n;i++)//打印格式
{
for(int j=1;j<=n;j++)
{
if(vis[i][j]==0)//已经搜到了
printf("2 ");
else
printf("%d ",map[i][j]);
}
printf("\n");
}
}
2.下午一直到傍晚(7h)
解决了马的遍历
题目描述
有一个 n \times mn×m 的棋盘,在某个点 (x, y)(x,y) 上有一个马,要求你计算出马到达棋盘上任意一个点最少要走几步。
输入格式
输入只有一行四个整数,分别为 n, m, x, yn,m,x,y。
输出格式
一个 n \times mn×m 的矩阵,代表马到达某个点最少要走几步(左对齐,宽 55 格,不能到达则输出 -1−1)。
输入输出样例
输入 #1复制
3 3 1 1
输出 #1复制
0 3 2 3 -1 1 2 1 4
说明/提示
数据规模与约定
对于全部的测试点,保证 1 \leq x \leq n \leq 4001≤x≤n≤400,1 \leq y \leq m \leq 4001≤y≤m≤400。
这道题我看了很久的题解才敲出去。。。
#include<cstdio>
#include<cstring>
#include<queue>
#include<iomanip>
using namespace std;
const int N=501;
int a[N][N];//定义501*501的储存答案的数组
struct point{
int x;
int y;
int t;
};//广搜结构体,对于每一个点,要储存它的横坐标(x坐标),纵坐标(y坐标),当前所走的步数
queue<point> que;//广搜必备队列,C++STL万岁O(∩_∩)O~~
int n,m,sx,sy;//棋盘边界、出发点坐标。
int dx[8]={-2,-2,-1,-1,2,2,1,1};
int dy[8]={1,-1,2,-2,1,-1,2,-2};//坐标偏移量
int main()
{
memset(a,-1,sizeof(a));//答案数组全部赋值-1,能达到就改为当前步数,不能改就直接输出。
scanf("%d%d%d%d",&n,&m,&sx,&sy);
a[sx][sy]=0;//切记,一定要把起点赋值为0,否则全WA
//广搜
que.push((point){sx,sy,0});//将起点放入队列
while(!que.empty())//只要还有可以走的点,就继续执行
{
point f=que.front();//将当前点拿出来
que.pop();//扔掉当前点
for(int i=0;i<=7;i++)//遍历当前点所能走到的其它点
{
int nx=f.x+dx[i];
int ny=f.y+dy[i];//通过坐标偏移量得到可以走的点的坐标
if(nx>=1&&nx<=n&&ny>=1&&ny<=m&&a[nx][ny]==-1)//当前点既没超出棋盘范围又还没有走过
{
a[nx][ny]=f.t+1;//这个点的答案就是上一个点的答案+1辣
que.push((point){nx,ny,f.t+1});//当前点可行,我们将其放入队列
}
}
}
//滑稽的输出
for(int i=1;i<=n;i++)
{
for(int j=1;j<=m;j++)
{
printf("%-5d",a[i][j]);
}
printf("\n");
}
return 0;//完美~^_^
}