0
点赞
收藏
分享

微信扫一扫

每日总结(2022/1/12)

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;//完美~^_^
}
举报

相关推荐

0 条评论