题意:有一个长度为n的字符串,每一位只会是p或j。你需要取出一个子串S(注意不是子序列),使得该子串不管是从左往右还是从右往左取,都保证每时每刻已取出的p的个数不小于j的个数。如果你的子串是最长的,那么称之为完美字符子串。求完美字符子串的长度。

乍一看比较水,然而差点没想出来。

考虑处理前缀和,p+1 j-1 那么对于一段区间[l,r]只要每一个i∈[l,r]满足presum[i]-presum[l-1]>=0即满足题意

1.单调队列预处理每个位置向前/后走的最远位置 记为f[i]/g[i]

2.枚举区间的左端点L,那么对于所有R∈[i,f[i]]找到最靠右的R满足g[R]<=i

3.R何可以通过向右移动cur优化,查找R可以用线段树优化

Code:

#include <bits/stdc++.h>
using namespace std; const int MAXN = 1000005; int n, f[MAXN], g[MAXN], sum1[MAXN], sum2[MAXN];
char s[MAXN];
int cnt, st[MAXN]; void solve1()
{
for(int i = 1; i <= n; i++)
if(s[i] == 'p') st[++cnt] = i;
else
{
while(cnt && sum1[st[cnt]] - sum1[i] > 1)
f[st[cnt--]] = i-1;
}
while(cnt) f[st[cnt--]] = n;
} void solve2()
{
for(int i = n; i >= 1; i--)
if(s[i] == 'p') st[++cnt] = i;
else
{
while(cnt && sum2[st[cnt]] - sum2[i] > 1)
g[st[cnt--]] = i+1;
}
while(cnt) g[st[cnt--]] = 1;
} int mn[MAXN*4]; void build(int i, int l, int r)
{
if(l == r)
{
mn[i] = (g[l] == 0) ? 0x7f7f7f7f : g[l];
return;
}
int mid = (l + r) >> 1;
build(i*2, l, mid);
build(i*2+1, mid+1, r);
mn[i] = min(mn[i*2], mn[i*2+1]);
} int find(int i, int l, int r, int x, int y, int k)
{
if(y < l || x > r || mn[i] > k) return -1;
if(l == r) return mn[i] <= k ? l : -1;
int mid = (l + r) >> 1;
int tmp = find(i*2+1, mid+1, r, x, y, k);
if(tmp != -1) return tmp;
return find(i*2, l, mid, x, y, k);
} int main ()
{
scanf("%d%s", &n, s+1);
for(int i = 1; i <= n; i++)
sum1[i] = sum1[i-1] + (s[i] == 'p' ? 1 : -1);
solve1();
for(int i = n; i >= 1; i--)
sum2[i] = sum2[i+1] + (s[i] == 'p' ? 1 : -1);
solve2();
build(1, 1, n); int cur = 1, Ans = 0;
for(int i = 1; i <= n; i++) if(f[i])
{
int pos = find(1, 1, n, cur=max(cur, i), f[i], i);
if(pos == -1) continue;
Ans = max(Ans, pos - i + 1); cur = pos+1;
}
printf("%d\n", Ans);
}

完美字符子串 单调队列预处理+DP线段树优化的更多相关文章

  1. [USACO2005][POJ3171]Cleaning Shifts(DP+线段树优化)

    题目:http://poj.org/problem?id=3171 题意:给你n个区间[a,b],每个区间都有一个费用c,要你用最小的费用覆盖区间[M,E] 分析:经典的区间覆盖问题,百度可以搜到这个 ...

  2. HDU4719-Oh My Holy FFF(DP线段树优化)

    Oh My Holy FFF Time Limit: 5000/2500 MS (Java/Others)    Memory Limit: 65535/65535 K (Java/Others) T ...

  3. UVA-1322 Minimizing Maximizer (DP+线段树优化)

    题目大意:给一个长度为n的区间,m条线段序列,找出这个序列的一个最短子序列,使得区间完全被覆盖. 题目分析:这道题不难想,定义状态dp(i)表示用前 i 条线段覆盖区间1~第 i 线段的右端点需要的最 ...

  4. zoj 3349 dp + 线段树优化

    题目:给出一个序列,找出一个最长的子序列,相邻的两个数的差在d以内. /* 线段树优化dp dp[i]表示前i个数的最长为多少,则dp[i]=max(dp[j]+1) abs(a[i]-a[j])&l ...

  5. 【uva1502/hdu4117-GRE Words】DP+线段树优化+AC自动机

    这题我的代码在hdu上AC,在uva上WA. 题意:按顺序输入n个串以及它的权值di,要求在其中选取一些串,前一个必须是后一个的子串.问d值的和最大是多少. (1≤n≤2×10^4 ,串的总长度< ...

  6. Contest20140906 ProblemA dp+线段树优化

    Problem A 内存限制 256MB 时间限制 5S 程序文件名 A.pas/A.c/A.cpp 输入文件 A.in 输出文件 A.out 你有一片荒地,为了方便讨论,我们将这片荒地看成一条直线, ...

  7. POJ 3171.Cleaning Shifts-区间覆盖最小花费-dp+线段树优化(单点更新、区间查询最值)

    Cleaning Shifts Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 4721   Accepted: 1593 D ...

  8. JZOJ 4738. 神在夏至祭降下了神谕 DP + 线段树优化

    4738. 神在夏至祭降下了神谕 Time Limits: 1000 ms  Memory Limits: 262144 KB  Detailed Limits   Goto ProblemSet D ...

  9. hdu3698 Let the light guide us dp+线段树优化

    http://acm.hdu.edu.cn/showproblem.php?pid=3698 Let the light guide us Time Limit: 5000/2000 MS (Java ...

随机推荐

  1. Storm Kafka与配置和代码集成

    1.目标 - 风暴卡夫卡整合 在本Kafka教程中,我们将学习Storm Kafka Integration的概念.此外,我们将在此Kafka Storm集成教程中讨论Storm架构,Storm Cl ...

  2. c++无关类型指针的强制转换原理的分析和尝试

    因最近看到大量的c类型指针强制转换,联系到c++的reinterpret_cast强制转换符,故总结一下. 先上图 由图中可以看出,先声明了一个结构体t与一个含有三个元素的数组num,接着声明一个指向 ...

  3. int main(int argc, char *argv[])解释

    int main(int argc, char *argv[]) 详解: #include <stdio.h> int main(int argc, char *argv[]) { int ...

  4. 基于node.js 的 websocket的移动端H5直播开发

    这一篇介绍一下基于node.js 的 websocket的移动端H5直播开发, 下载文章底部的源码,我是用vscode打开, 首先在第一个终端运行 npm run http-server 这个指令是运 ...

  5. Java字节流文件复制

    1.字节流 在 Java 中,文件的复制使用字节输入流和字节输出流实现,java.io 包有 InputStream 和 OutputStream 这两个顶层抽象类规范了读写文件所需的核心 API. ...

  6. 【洛谷 P3193】 [HNOI2008]GT考试(KMP,dp,矩阵乘法)

    题目链接 \(f[i][j]\)表示准考证号到第\(i\)位,不吉利数字匹配到第\(j\)位的方案数. 答案显然是\(\sum_{i=0}^{m-1}f[n][i]\) \(f[i][j]=\sum_ ...

  7. Js保存图片到本地

    注:此方法是使用hbuilderx云打包之后才能用,否则在浏览器中会报 plus is not defined 官方文档 http://www.html5plus.org/doc/zh_cn/gall ...

  8. requests模块 简单使用

    目录 requests模块 简单使用 Anaconda简单了解 简单了解 requests模块 使用requests模块 爬取搜狗首页源码数据 实现一个简易的网页采集器 解决乱码问题 解决UA检测问题 ...

  9. vue routes路由

    mode: 'history',去掉浏览器上url前的#号

  10. 微信支付接口--超详细带注释代码--Demo

    如果本文对你有用,请爱心点个赞,提高排名,帮助更多的人.谢谢大家!❤ 如果解决不了,可以在文末进群交流. 如果对你有帮助的话麻烦点个[推荐]~最好还可以follow一下我的GitHub~感谢观看! 微 ...