PAT (1003) 我要通过! 20分

题目传送门:我要通过!
首先思考题目给的三个重要信息:

  • 1.字符串中必须仅有 P、 A、 T这三种字符,不可以包含其它字符;
  • 2.任意形如 xPATx 的字符串都可以获得“答案正确”,其中 x 或者是空字符串,或者是仅由字母 A 组成的字符串;
  • 3.如果 aPbTc 是正确的,那么 aPbATca 也是正确的,其中 a、 b、 c 均或者是空字符串,或者是仅由字母 A 组成的字符串。
    思路
  • 根据第一个条件得出该字符串必须包含PAT这三个字符,而且不含其他字符。
  • 再来看第二个条件,可以得出字符串主体“PAT“两侧的是对称空或含A字符串
  • 最后分析条件三,若aPbTc正确,那么aPbATca也正确。很明显两个字符串中唯一没变的是字符串主体PAT左侧的a串,若a串为空,那么c也必须为空,这样推下去所有形如PA……(n个A)T,n为任意正整数,这样的字符串也是正确的。那么如果a串不是空串而是只含有A的字符串的情况呢,显然,aPbTc若正确,由条件2可得出,由条件3而衍生的字符串中的母串必须形如xPATx。我们可以假设a=c,b=A。那么在此规律衍生下去的字符串中c中A的个数必须为a中A的个数的n倍,则b中A的个数为n。
  • 由此规律我们可以统计P左侧的A个数s1,P和T中间的A个数s2,T右侧的A个数s3然后得出s3/s1=s2.
    废话少说,上代码
#include<algorithm>
#include<iostream>
#include<queue>
#include<vector>
#include<string>
#include<utility>
#include<stack>
#include<map>
#include<set>
#include<iomanip>
#include<cmath>
#include<cstdio>
#include<cstdlib>
#include<cstring>
#define N 110
#define mod 80112002
#define INF 0x3f3f3f3f
#define ll long long
#define Pii pair<int,int>
#define Pdd pair<double,double>
#define Pll pair<ll,ll>
using namespace std;
int n;
string s;
int main(){
    ios::sync_with_stdio(false);
    cin>>n;
    while(n--){
        int p=0,a=0,t=0;
        bool flag=true;
        cin>>s;
        int len=s.length();
        for(int i=1;i<=len;i++){
            char ch=s[i-1];
            if(ch=='P') p=i;
            else if(ch=='T') t=i;
            else if(ch=='A') a++;
            else{
                flag=false;
                break;
            }
        }
        if(p&&a&&t&&p<t){
            if(p-1&&len-t){
                if((p-1)==(len-t)){
                    for(int i=1;i<=p-1;i++){
                        if(s[i-1]!='A'){
                            flag=false;
                            break;
                        }
                    }
                    for(int i=t+1;i<=len;i++){
                        if(s[i-1]!='A'){
                            flag=false;
                            break;
                        }
                    }
                    int sm=0;
                    for(int i=p+1;i<=t-1;i++){
                        if(s[i-1]=='A') sm++;
                    }
                    if(sm!=1) flag=false;
                }
                else{
                    int s1=0,s2=0,s3=0;
                    for(int i=1;i<=p-1;i++){
                        if(s[i-1]=='A') s1++;
                    }
                    for(int i=p+1;i<=t-1;i++){
                        if(s[i-1]=='A') s2++;
                    }
                    for(int i=t+1;i<=len;i++){
                        if(s[i-1]=='A') s3++;
                    }
                    if(s1&&s3){
                        if(s3%s1) flag=false;
                        else{
                            int ans=s3/s1;
                            if(s2!=ans) flag=false;
                        }
                    }
                }
            }
            else{
                for(int i=p+1;i<=t-1;i++){
                    if(s[i-1]!='A'){
                        flag=false;
                        break;
                    }
                }
            }
        }
        else flag=false;
        if(flag) cout<<"YES\n";
        else cout<<"NO\n";
    }
    return 0;
}

在这里插入图片描述
然后就可以快乐ac啦=_=

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值