【BZOJ】3521: [Poi2014]Salad Bar
题意
长度为\(n(1 \le n \le 1000000)\)的\(01\)字符串。找一个最长的连续子串\(S\),使得不管是从左往右还是从右往左取,都保证每时每刻已取出的\(1\)的个数不小于\(0\)的个数。
分析
首先对\(i\)求出\(l_i, r_i\),\(l_i\)表示在区间\([l_i, i]\)从左往右一直取,\(1\)的个数总是不少于\(0\)的个数的最远\(l_i\)。\(r_i\)同理。由于前缀和前后差绝对值为\(1\),所以我们可以开一个数组在\(O(n)\)内求出这\(l_i, r_i\)(比如求\(l_i\)就是我们只需要找一个满足\(sum_i+1=sum_j, j < i\)的最大\(j\))
然后我们对\(l_i\)升序排序(假设得到的顺序是\(b_i\))。然后从左到右扫过去,当前在\(i\)位置,每次更新满足\(l_{b_j} \le i\)的所有的\(b_j\)到区间\([b_j, n]\),然后对于\(i\),则向右能拓展到的位置就是区间\([1, i]\)的最大值。
题解
维护这个前缀最大值用个\(bit\)就行了。复杂度\(O(nlogn)\)。
#include <bits/stdc++.h>
using namespace std;
const int N=1000005;
int n, a[N], b[N+N], c[N+N], s1[N], s2[N], l[N], r[N], mx[N];
void add(int x, int g) {
for(; x<=n; x+=x&-x) {
mx[x]=max(mx[x], g);
}
}
int getmx(int x) {
int y=0;
for(; x; x-=x&-x) {
y=max(y, mx[x]);
}
return y;
}
void isort(int *y, int *x) {
for(int i=0; i<=n; ++i) c[i]=0;
for(int i=1; i<=n; ++i) c[y[i]]++;
for(int i=1; i<=n; ++i) c[i]+=c[i-1];
for(int i=1; i<=n; ++i) x[c[y[i]]--]=i;
}
int main() {
scanf("%d\n", &n);
for(int i=1; i<=n; ++i) {
a[i]=getchar()=='p'?1:-1;
}
for(int i=1, j=n; i<=n; ++i, --j) {
s1[i]=s1[i-1]+a[i];
s2[j]=s2[j+1]+a[j];
b[s1[i]+N]=b[s1[i]+1+N]=-1;
c[s2[j]+N]=c[s2[j]+1+N]=n+2;
}
for(int i=1, j=n; i<=n; ++i, --j) {
if(a[i]==1) {
l[i]=b[s1[i]+1+N]+2;
}
if(a[j]==1) {
r[j]=c[s2[j]+1+N]-2;
}
b[s1[i]+N]=i;
c[s2[j]+N]=j;
}
isort(l, b);
int nl=1, t, ans=0;
for(int i=1; i<=n; ++i) {
while(nl<=n && l[t=b[nl]]<=i) {
if(a[t]!=-1) {
add(t, t);
}
++nl;
}
if(a[i]!=-1) {
ans=max(ans, getmx(r[i])-i+1);
}
}
printf("%d\n", ans);
return 0;
}
【BZOJ】3521: [Poi2014]Salad Bar的更多相关文章
- 【BZOJ】3524: [Poi2014]Couriers
[算法]主席树 [题解]例题,记录和,数字出现超过一半就递归查找. 主席树见[算法]数据结构 #include<cstdio> #include<algorithm> #inc ...
- 【BZOJ】3835: [Poi2014]Supercomputer
题意 \(n(1 \le 1000000)\)个点的有根树,\(1\)号点为根,\(q(1 \le 1000000)\)次询问,每次给一个\(k\),每一次可以选择\(k\)个未访问的点,且父亲是访问 ...
- 【BZOJ】3832: [Poi2014]Rally
题意 \(n(2 \le n \le 500000)\)个点\(m(1 \le m \le 1000000)\)条边的有向无环图,找到一个点,使得删掉这个点后剩余图中的最长路径最短. 分析 神题不会做 ...
- 【BZOJ】3526: [Poi2014]Card
题意 \(n(n \le 200000)\)张卡片,正反有两个数\(a[i], b[i]\).\(m(m \le 1000000)\)次操作,每次交换\(c[i].d[i]\)位置上的卡片.每一次操作 ...
- 【BZOJ】3523: [Poi2014]Bricks
题意 \(n(n \le 1000000)\)个物品,颜色分别为\(a[i]\),现在要求排在一排使得相邻两个砖块的颜色不同,且限定第一个砖块和最后一个砖块的颜色,输出一个合法解否则输出-1. 分析 ...
- 【BZOJ】3834: [Poi2014]Solar Panels
http://www.lydsy.com/JudgeOnline/problem.php?id=3834 题意:求$max\{(i,j)\}, smin<=i<=smax, wmin< ...
- 【BZOJ】3524 [POI2014] Couriers(主席树)
题目 传送门:QWQ 传送到洛谷QWQ 分析 把求区间第k大的改一改就ok了. 代码 #include <bits/stdc++.h> using namespace std; ; ], ...
- 【BZOJ】3052: [wc2013]糖果公园
http://www.lydsy.com/JudgeOnline/problem.php?id=3052 题意:n个带颜色的点(m种),q次询问,每次询问x到y的路径上sum{w[次数]*v[颜色]} ...
- 【BZOJ】3319: 黑白树
http://www.lydsy.com/JudgeOnline/problem.php?id=3319 题意:给一棵n节点的树(n<=1e6),m个操作(m<=1e6),每次操作有两种: ...
随机推荐
- ubuntu sudo update与upgrade的作用及区别
ubuntu sudo update与upgrade的作用及区别 入门linux的同志,刚开始最迫切想知道的,大概一个是中文输入法,另一个就是怎么安装软件.本文主要讲一下LINUX安装软件方面的特点. ...
- DD_belatedPNG.js解决透明PNG图片背景灰色问题
<!--[]> <script type="text/javascript" src="http://www.phpddt.com/usr/themes ...
- Android检测网络是否正常代码!
在Android开发中,如果该应用程序需要连接网络请求,那么最好我们先做一个检测网络是否在线的判断,否则程序容易出现卡死或FC等Bug,应该判断如果手机离线则弹出提示让用户检查网络,如果正常则继续执行 ...
- Swift3.0P1 语法指南——类和结构体
原档:https://developer.apple.com/library/prerelease/ios/documentation/Swift/Conceptual/Swift_Programmi ...
- 【Alpha版本】冲刺-Day5
队伍:606notconnected 会议时间:11月13日 会议总结 张斯巍(433) 今天安排:完成昨天没完成的,设置界面设计 完成度:85% 明天计划:学习UI设计 遇到的问题:无 感想:一定要 ...
- javascript数据结构-优先队列
这里之所以扩充一个 有限队列 是因为,生活使用中队列通常会附加优先级,比如排队买票,一般老人和军人等会有优先权限. 实现:继承上篇的 普通队列实现.这里用一种方法,入队的时候,进行排序插入到指定位置, ...
- 在vs中char类型的实参与LPCWSTR类型的形参类型不兼容怎么解决?
今天在做 COS脚本解释器的时候,遇到了这个问题 先了解一下 LPCTCHAR 这个东东 LPCTSTR用来表示你的字符是否使用UNICODE, 如果你的程序定义了UNICODE或者其他相关的宏,那么 ...
- Android 双击 Back 键退出程序
双击退出程序的原理无非就是设置一个退出标识(询问是否退出),如果改变了这个标识(确认退出),则再次点击时立马退出,如果短时间内没有退出,则延时重置这个标识(不退出). ================ ...
- SQLite批量插入优化方法
SQLite的数据库本质上来讲就是一个磁盘上的文件,所以一切的数据库操作其实都会转化为对文件的操作,而频繁的文件操作将会是一个很好时的过程,会极大地影响数据库存取的速度. 例如:向数据库中插入10 ...
- Oracle分析函数(一)
一.总体介绍 分析函数如何工作 语法 FUNCTION_NAME(<参数>,…) OVER (<PARTITION BY 表达式,…> <ORDER BY 表达式 < ...