Processing math: 100%

「模拟赛20191019」A 简单DP

题目描述

给一个n×m的网格,每个格子上有一个小写字母。

对于所有从左上角(1,1)到右下角(n,m)只向下或向右走的路径构成的集合,判断是否存在两条走法不同的路径,使得把它们经过的格子上的字母按顺序记下来得到的序列完全相同。

输入

第一行一个整数T表示数据组数,对于每组数据:
第一行两个整数n,m
接下来n行,每行一个长度为m的字符串表示网格的每一行。

输出

对于每组数据,输出一行YesNo表示是否存在这样的两条路径。

样例

样例输入

2
2 2
ab
bc
2 2
ab
cd

样例输出

Yes
No

数据范围

对于100%的数据,1n,m10001T10

水题,考虑DP,假设目前考虑到第i行,第j列,要是存在这两条路径,那么它们要么都从上方走来,要么都从左边走来,或者一条从上方走来,一条从左边走来。

讨论一下,第一种情况,只要i1行,j列可以满足就行了;第二种情况,只要i行,j1列满足就行了;第三种情况,首先要满足Ai1,j=Ai,j1,然后发现只要能走到(i1,j1)这个点,就一定存在这样的路径(先沿着同一条路走到(i1,j1),然后分别经过(i1,j)(i,j1)到达(i,j))。

于是定义fi,j表示前i行,j列是否满足,则fi,j=fi1,j|fi,j1|(Ai1,j=Ai,j1)。这道题就做完了……

Code:

#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
using namespace std;
#define N 1005
char S[N][N];
int n, m, t, f[N][N];
int main()
{
	scanf("%d", &t);
	for (; t--; )
	{
		scanf("%d%d", &n, &m);
		for (int i = 1; i <= n; i++)
			scanf("%s", S[i] + 1);
		memset(f, 0, sizeof(f));
		for (int i = 1; i <= n; i++)
			for (int j = 1; j <= m; j++)
			{
				f[i][j] |= f[i - 1][j] | f[i][j - 1];
				if (i > 1 && j > 1)
					f[i][j] |= (S[i][j - 1] == S[i - 1][j]);
			}
		if (f[n][m])
			puts("Yes");
		else
			puts("No");
	}
}
posted @   ModestStarlight  阅读(219)  评论(0)    收藏  举报
编辑推荐:
· 记一次 .NET某旅行社酒店管理系统 卡死分析
· 长文讲解 MCP 和案例实战
· Hangfire Redis 实现秒级定时任务,使用 CQRS 实现动态执行代码
· Android编译时动态插入代码原理与实践
· 解锁.NET 9性能优化黑科技:从内存管理到Web性能的最全指南
阅读排行:
· 一天 Star 破万的开源项目「GitHub 热点速览」
· 瞧瞧别人家的日期处理,那叫一个优雅!
· 使用TypeScript开发微信小程序(云开发)-入门篇
· 没几个人需要了解的JDK知识,我却花了3天时间研究
· 定时任务稳定性解决方案-healthchecks监控系统
点击右上角即可分享
微信分享提示