• QDUOJ 河老师的新年礼物(尺取法)


    河老师的新年礼物

    发布时间: 2017年1月1日 15:11   最后更新: 2017年1月1日 15:13   时间限制: 1000ms   内存限制: 256M

    河老师的新年礼物是一个长度为n的ab串,他想要找出最长的一个子串使得这个子串中每个字符都相等,他称之为“最优子串”。当然对河老师来说这个问题太简单了,于是他加了一个条件:他可以改变这个串中的某些字符,但一次只能改变一个字符,最多能改变k次。河老师想要知道,在可以对串进行改变的前提下,这个ab串的“最优子串”的长度是多少。

    第一行是一个整数T,代表T组测试数据,
    第二行是两个整数 n , k (1 ≤ n ≤ 100 000, 0 ≤ k ≤ n) 
    第三一个ab串,长度为n

    输出“最优子串”的长度

     复制
    2
    4 2
    abba
    8 1
    aabaabaa
    4
    5

    给你一串字符串和k个修改字符串的机会,让这个字符串获得最长连续相同子串

    题解:所谓尺取法,顾名思义,就是一把尺子(固定某一条件),不断向右(向左)移动,不断更新我们要的答案。在这里,我们只要从左往右,让修改的字符个数从0慢慢增加到k,中途将字符改成同一个字符(a改成b或者b改成a都行),最后修改字符数固定为k,每次向右移动时,如果字符串需要修改,那就改掉右面的字符,将之前最左边的字符换回来。那么我们可以用一个队列去实现。

    如果你看不懂上面也没关系,我再讲具体一些。我们从左边开始,扫描字符串。如果遇到a,那就把a丢进队列,如果遇到b,且此时队列中b的个数不超过k的话,就把b丢进去。如果b的个数等于k,且又遇到一个b,那就记录下此时队列长度,这个时候队列里k个b可以当作a,所以可以记录队列元素个数,更新答案最大值。之后将队列前面元素弹出,直到弹出一个b,再将新的b压进队列……一直扫描直到字符串尾。

    接下来将a和b互换,重复上面步骤,更新最大值。最后输出最大值。

    第一次写的队列是存取了修改字符的下标,更新最大值为当前下标与弹出下标的差。在codeforces上能过,qduoj就WA,这种方法效率会比上面的高一些,毕竟队列里存入的只是部分值,还特地考虑了k=0的情况,应该还是数据(shui)的问题吧>_<

    #include<stdio.h>
    #include<string.h>
    #include<queue>
    using namespace std;
    
    int main()
    {
        int t,c,n,k,maxa,maxb,max,i;
        char s[100005];
        queue<int> q;
        scanf("%d",&t);
        while(t--){
            scanf("%d%d",&n,&k);
            scanf(" %s",s);
            max=0;
            while(q.size()) q.pop();
            maxa=0;c=0;
            for(i=1;i<=n;i++){
                if(s[i-1]=='a'){
                    q.push(1);
                }
                if(s[i-1]=='b'){
                    q.push(2);
                    if(c<k) c++;
                    else{
                        while(q.size()&&q.front()==1) q.pop();
                        if(q.size()) q.pop();
                    }
                }
                if(q.size()>maxa) maxa=q.size();
            }
            while(q.size()) q.pop();
            maxb=0;c=0;
            for(i=1;i<=n;i++){
                if(s[i-1]=='b'){
                    q.push(2);
                }
                if(s[i-1]=='a'){
                    q.push(1);
                    if(c<k) c++;
                    else{
                        while(q.size()&&q.front()==2) q.pop();
                        if(q.size()) q.pop();
                    }
                }
                if(q.size()>maxb) maxb=q.size();
            }
            if(maxa>maxb) max=maxa;
            else max=maxb;
            printf("%d
    ",max);
        }
        return 0;
    }
  • 相关阅读:
    lua 语言基础
    C语言基础
    文件夹目录排序
    C#调用Server_SQL
    SQL语言基础
    批量修改文件名
    快捷键Alt、Shift、Ctrl 点击事件
    第一节:基础语法
    一:ASP.NET基础知识(二)
    孩子,我首先希望你自始至终都是一个理想主义者!
  • 原文地址:https://www.cnblogs.com/yzm10/p/7392694.html
Copyright © 2020-2023  润新知