PAT Basic 1003 “我要通过!”

PAT乙级1003 “我要通过!” C语言题解

题设

​ “答案正确”是自动判题系统给出的最令人欢喜的回复。本题属于 PAT 的“答案正确”大派送 —— 只要读入的字符串满足下列条件,系统就输出“答案正确”,否则输出“答案错误”。

得到“答案正确”的条件是:

  1. 字符串中必须仅有 P、 A、 T这三种字符,不可以包含其它字符;
  2. 任意形如 xPATx 的字符串都可以获得“答案正确”,其中 x 或者是空字符串,或者是仅由字母 A 组成的字符串;
  3. 如果 aPbTc 是正确的,那么 aPbATca 也是正确的,其中 a、 b、 c 均或者是空字符串,或者是仅由字母 A 组成的字符串。

现在就请你为 PAT 写一个自动裁判程序,判定哪些字符串是可以获得“答案正确”的。

输入格式:
每个测试输入包含 1 个测试用例。第 1 行给出一个正整数 n (<10),是需要检测的字符串个数。接下来每个字符串占一行,字符串长度不超过 100,且不包含空格。

输出格式:
每个字符串的检测结果占一行,如果该字符串可以获得“答案正确”,则输出 YES,否则输出 NO。

//输入样例:
8
PAT
PAAT
AAPATAA
AAPAATAAAA
xPATx
PT
Whatever
APAAATAA

//输出样例:
YES
YES
YES
YES
NO
NO
NO
NO

题解

​ 一开始单纯以为是个字符串分割 + 递归的题,结果用Java写有一个用例一直通不过,**后来看了答案发现原来纯粹是个数学问题。**原来是没有读懂题,读懂了题目想要表达的意思就很简单了。

​ 结合条件1和2,容易得到,字符串只能含有字符 P A T三者之一,而且P和T只能出现一次,P必须在T的前面。

注意x,a,b,c都是字符串, 我们可以想象 P 和 T把传入的字符串分割成了三个部分。

​ 条件2是最简单的一种情况,xPATx。

​ 条件3实际上是一个递归嵌套的定义,若aPbTc正确, 则 aPbATca 正确。 从后者到前者的转换是: P和T之间的bA子串去掉一个A字符,T后的ca部分去掉a。 也就是说, P和T之间的部分每去掉一个A字符,T后的部分就去掉一个a子串 , 重复这样的过程,直到转换成 条件2的 xPATx 这样的情况(YES)

​ 我们把传入的字符串用 m*A P n*A T k*A 代替,m,n,k分别表示三部分所含A字符的个数

​ 那么满足条件的字符串有:n>0 且 k = n*m

​ 只需要遍历一次字符串就可以把m,n,k算出来啦

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
int check(char str[]){
	int m = 0,n = 0,k = 0;
	int P_pos = -1, T_pos = -1;	//分别代表字符P的索引,字符T的索引
	int T_flag = 0;	// 1:->已经读到了T字符,0->还未读到T字符
	for(int i = 0 ; i < strlen(str); i++){
		char ch = str[i];
		if(ch != 'P' && ch != 'A' && ch!='T'){ // 只能是PAT三者之一
			return 0;
		}
       
		if(ch == 'P'){	
            // P只能出现一次
			if(P_pos != -1){	
				return 0;
			}
			P_pos = i;	
		}
		if(ch == 'T'){
            // T只能出现一次
			if(T_pos != -1){
				return 0;
			}
			T_pos = i;
			T_flag = 1; // T已经读到了
		}	
		// P必须在T前面
		if(T_pos > P_pos && T_flag != 1){
			return 0;
		}
		
	}
	
	m = P_pos;
	n = T_pos - P_pos - 1;
	k = strlen(str) - T_pos - 1;
	return n > 0 && k == n * m;		//n > 0 是为了排除 "PT"这样的例子

}

int main(int argc, char *argv[]) {
	
	int n=0;
	scanf("%d",&n);
	char str[101];
	while(n--){
		scanf("%s",&str);
		if(check(str)){
			printf("YES\n");
		}else{
			printf("NO\n");
		}
	}
	return 0;
}
测试点结果耗时内存
0答案正确2 ms296 KB
1答案正确2 ms296 KB
2答案正确2 ms376 KB
3答案正确2 ms256 KB
4答案正确2 ms256 KB
5答案正确2 ms128 KB
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值