HRZ学英语(类似尺取)

时间限制1s,空间限制64MB

题目描述

瑞神今年大三了,他在寒假学会了英文的26个字母,所以他很兴奋!
于是他让他的朋友TT考考他,TT想到了一个考瑞神的好问题:给定一个字符串,从里面寻找 连续的26个大写字母 并输出!
但是转念一想,这样太便宜瑞神了,所以他加大了难度:现在给定一个字符串,字符串中包括26个大写字母和特殊字符’?’,特殊字符’?'可以代表任何一个大写字母。
现在TT问你是否存在一个 位置连续的且由26个大写字母组成的子串 ,在这个子串中每个字母出现且仅出现一次,如果存在,请输出从左侧算起的第一个出现的符合要求的子串,并且要求,如果有多组解同时符合位置最靠左,则输出字典序最小的那个解!如果不存在,输出-1!
这下HRZ蒙圈了,他刚学会26个字母,这对他来说太难了,所以他来求助你,请你帮他解决这个问题,报酬是可以帮你打守望先锋。
说明:字典序 先按照第一个字母,以 A、B、C……Z 的顺序排列;如果第一个字母一样,那么比较第二个、第三个乃至后面的字母。如果比到最后两个单词不一样长(比如,SIGH 和 SIGHT),那么把短者排在前。例如

AB??EFGHIJKLMNOPQRSTUVWXYZ

ABCDEFGHIJKLMNOPQRSTUVWXYZ
ABDCEFGHIJKLMNOPQRSTUVWXYZ

上面两种填法,都可以构成26个字母,但是我们要求字典序最小,只能取前者。
注意,题目要求的是 第一个出现的, 字典序最小的 !

Input

输入只有一行,一个符合题目描述的字符串。

Output

输出只有一行,如果存在这样的子串,请输出,否则输出-1

Sample intput & output

Sample input1
ABC??FGHIJK???OPQR?TUVWXY?
Sample output1
ABCDEFGHIJKLMNOPQRSTUVWXYZ
Sample input2
AABCDEFGHIJKLMNOPQRSTUVW??M
Sample output2
-1

数据范围

在这里插入图片描述

解题思路

类似于尺取的一种区间移动问题,就是如果当前 [ l , r ] [l,r] 区间满足,则输出结果,如果不满足,则 l + + , r + + l++,r++ 。更多讲解见代码。

完整代码

//#pragma GCC optimize(2)//比赛禁止使用!
//#pragma G++ optimize(2)
//#include <bits/stdc++.h>
#include <iostream>
#include <cstdio>
#include <cstdlib>
#include <cmath>
#include <cstring>
#include <string>
#include <climits>
#include <algorithm>
#include <queue>
#include <vector>
using namespace std;

int ch[100],l,r;
char zero[100];
string s;
bool check()//检查是否满足情况
{
    int temp=0;
    for (int i=0; i<26; i++){
        if(ch[i]>1) return false;//如果一个字符出现超过了1次,不满足
        else if(ch[i]==0) temp++;//没有出现的字符数量如果最终等于?的数量就满足
    }
    return temp==ch[26];
}
void print()
{
    int temp=0;
    for (int i=0; i<26; i++)
        if(ch[i]==0)
            zero[++temp]=char(i+'A');//将没有出现过的字符存进来,便于后面输出的时候是字典序最小的
    temp=0;
    for (int i=l; i<=r; i++)
    {
        if(s[i]!='?') cout<<s[i];
        else cout<<zero[++temp];
    }
    cout<<endl;
}
int main()
{
    cin>>s;
    if(s.size()<26) { cout<<-1<<endl; return 0;}
    for (int i=0; i<26; i++)//预先处理前26个字符,ch[0-25]存'A'-'Z',ch[26]存'?'
    {
        if(s[i]!='?') ch[s[i]-'A']++;
        else ch[26]++;
    }
    l=0,r=25;
    while(r<s.size()){
        if(check()){ print(); return 0; }
        else {
            if(s[l]!='?') ch[s[l]-'A']--;//左端点右移前,将这个位置的字符去掉
            else ch[26]--;
            l++; r++;
            if(r<s.size()) {//右端点右移后,将新加入的字符计算入ch数组
                if(s[r]!='?') ch[s[r]-'A']++;
                else ch[26]++;
            }
        }
    }
    cout<<-1<<endl;
    return 0;
}
发布了45 篇原创文章 · 获赞 39 · 访问量 4534

猜你喜欢

转载自blog.csdn.net/weixin_43347376/article/details/105387136