题解 [牛客IOI周赛16-提高组 B] 小L扔垃圾

小L扔垃圾

https://ac.nowcoder.com/acm/contest/5388/B

题目描述

给定一个序列,每次在其左端点或者右端点加入一个元素。

求这个序列的最长子段满足这个子段的 'W' 元素的个数等于 'D' 元素的个数。

正解

把 'W' 看成 1,把 'D' 看成 -1,若一个子段满足条件,则这个子段的和为 0。

记录前缀和,然后 要满足条件的话,

离线预处理出前缀和数组,用哈希表维护一个前缀和的出现位置就行。

时间复杂度

(貌似在线也能做

代码

#include <bits/stdc++.h>
#define N 2000005

using namespace std;

int n;
int arr[N + N], ltax[N + N], rtax[N + N], dir[N];
char s[N];

unordered_map<int, int> l, r;


int get(char ch = 0) {
    while(ch != 'R' && ch != 'W' && ch != 'D') ch = getchar();
    if(ch == 'R') return 0;
    return ch == 'W' ? -1 : 1;
}

int main() {
    int *f = arr + N, *psl = ltax + N, *psr = rtax + N;
    int lp = 1, rp = 0, ans = 0;
    int lsum = 0, rsum = 0;

    scanf("%d", &n);
    scanf("%s", s + 1);
    int len = strlen(s + 1);
    for(int i = 1; i <= len; ++i)
        f[++rp] = get(s[i]);

    for(int i = 1; i <= n; ++i) {
        scanf("%d", &dir[i]);
        if(dir[i] == 1) {
            f[--lp] = get();
        } else {
            f[++rp] = get();
        }
    }

    for(int i = lp; i <= rp; ++i) psr[i] = psr[i - 1] + f[i];
    for(int i = rp; i >= lp; --i) psl[i] = psl[i + 1] + f[i];

    lp = 1, rp = 0;
    r[psr[rp]] = rp;
    for(int i = 1; i <= len; ++i) {
        ++rp;
        if(!r.count(psr[rp])) {
            r[psr[rp]] = rp;
        } else {
            ans = max(ans, rp - r[psr[rp]]);
        }
    }
    for(int i = 1; i <= len; ++i) l[psl[i]] = i;
    l[psl[len + 1]] = len + 1;
    printf("%d\n", ans);

    for(int i = 1; i <= n; ++i) {
        if(dir[i] == 1) {
            --lp, r[psr[lp - 1]] = lp - 1;
            if(!l.count(psl[lp])) {
                l[psl[lp]] = lp;
            } else {
                ans = max(ans, l[psl[lp]] - lp);
            }
        } else {
            ++rp, l[psl[rp + 1]] = rp + 1;
            if(!r.count(psr[rp])) {
                r[psr[rp]] = rp;
            } else {
                ans = max(ans, rp - r[psr[rp]]);
            }
        }
        printf("%d\n", ans);
    }
    return 0;
}
全部评论

相关推荐

06-12 16:23
已编辑
小米_软件开发(准入职员工)
点赞 评论 收藏
分享
不愿透露姓名的神秘牛友
07-11 12:31
以前小时候我最痛恨出轨、偷情的人,无论男女,为什么会出轨?现在我成了自己最讨厌的人,没想到分享的东西在牛客会被这么多人看,大家的评价都很中肯,我也认同,想过一一回复,但我还是收声了,我想我应该说说这件事,这件事一直压在我心里,是个很大的心结,上面说了人为什么出轨,我大概能明白了。我们大一下半年开始恋爱,开始恋爱,我给出了我铭记3年的承诺,我对她好一辈子,我永远不会背叛,我责任心太重,我觉得跟了我,我就要照顾她一辈子,我们在一起3年我都没有碰过她,她说往东我就往东,她说什么我做什么,她要我干什么,我就干什么!在学校很美好,中途也出过一些小插曲,比如男闺蜜、男闺蜜2号等等等。但我都强迫她改掉了,我...
牛客刘北:两个缺爱的人是没有办法好好在一起的,但世界上哪有什么是非对错?你后悔你们在一起了,但是刚刚在一起的美好也是真的呀,因为其他人的出现,你开始想要了最开始的自己,你的确对不起自己,21岁的你望高物远,你完全可以不谈恋爱,去过你想要的生活,你向往自由,在一起之后,你要想的不是一个人,而是两个人,你不是变心了,就像你说的,你受够了,你不想包容了,冷静几天是你最优的选择,爱人先爱己。
社会教会你的第一课
点赞 评论 收藏
分享
评论
1
收藏
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务