洛谷1101 单词方阵

该博客介绍了洛谷1101题目的详细内容,涉及一个nXn的字母方阵,其中包含多个沿特定方向排列的单词。题目要求将非单词的字母替换为'*',以突出显示单词。解决方案提到使用DFS递归方法,并通过标记满足条件的字符坐标来处理可能的交叉情况。博主详细解释了如何遍历和标记这些字符,以实现题目要求的输出格式。
摘要由CSDN通过智能技术生成

题目描述
给一nXn的字母方阵,内可能蕴含多个“yizhong”单词。单词在方阵中是沿着同一方向连续摆放的。摆放可沿着8个方向的任一方向,同一单词摆放时不再改变方向,单词与单词之间可以交叉,因此有可能共用字母。输出时,将不是单词的字母用“*”代替,以突出显示单词。例如:

输入:
8                     输出:
qyizhong              *yizhong
gydthkjy              gy******
nwidghji              n*i*****
orbzsfgz              o**z****
hhgrhwth              h***h***
zzzzzozo              z****o**
iwdfrgng              i*****n*
yyyygggg              y******g
输入输出格式
输入格式:
第一行输入一个数n。(7<=n<=100)。

第二行开始输入nXn的字母矩阵。

输出格式:
突出显示单词的nXn矩阵。

DFS递归解决,用一个二维字符数组接受输入。注意到可以有交叉,故需要把所有满足要求的字符的坐标标记,输出时除标记点外均输出*即可。对每一个符合要求的字符循环8个方向进行遍历即可,如果该方向符合要求,则将该方向的7个坐标标记。

#include<iostream>
using namespace std;
int n, wdl = 0;
int dx[8] = { -1,1,0,0,-1,1,1,-1 }, dy[8] = { 0,0,-1,1,-1,-1,1,1 };  //8个方向
char s[150][150];                                                    
int flag[150][150] = { 0 };               //标记数组
char ss[7] = { 'y','i','z','h','o','n','g' }; 
int memory[14], c = 2;                    //memory临时记忆数组,记录该方向的点的坐标
void dfs(int x, int y,int i,int num)      //xy是当前点的坐标,i指示当前的方向,num记录个数
{
	if (num == 7)                         //当num为7是说明符合要求,标记一下
		wdl = 1;
	else if (x + dx[i] >= 1 && x + dx[i] <= n&&y + dy[i] >= 1 && y + dy[i] <= n&&s[x + dx[i]][y + dy[i]] == ss[num])             
	{
		memory[c++] = x + dx[i];          //如果下一个点符合则继续在下一个点处递归
		memory[c++] = y + dy[i];
		dfs(x + dx[i], y + dy[i], i, num + 1);
	}
	else   wdl = 0;                       //该方向不符合要求
}
int main()
{
	cin >> n;
	for (int i = 1; i <= n; i++)
		for (int j = 1; j <= n; j++)
			cin >> s[i][j];
	for (int i = 1; i <= n; i++)
	{
		for (int j = 1; j <= n; j++)
		{
			if (s[i][j] == 'y')
			{
				for (int k = 0; k < 8; k++)
				{
					memset(memory, 0, sizeof(memory));     //清空重置!
					wdl = 0, c = 2;
					memory[0] = i, memory[1] = j;          //先记录初始点坐标
					dfs(i, j, k, 1);
					if (wdl == 1)
						for (int l = 0; l < 14; l += 2)
							flag[memory[l]][memory[l + 1]] = 1;
				}
			}
		}
	}
	for (int i = 1; i <= n; i++)
	{
		for (int j = 1; j <= n; j++)
		{
			if (flag[i][j])
				cout << s[i][j];
			else
				cout << '*';
		}
		cout << endl;
	}
	return 0;
}

 

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值