题意

长度为\(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)\)。

  1. #include <bits/stdc++.h>
  2. using namespace std;
  3. const int N=1000005;
  4. int n, a[N], b[N+N], c[N+N], s1[N], s2[N], l[N], r[N], mx[N];
  5. void add(int x, int g) {
  6. for(; x<=n; x+=x&-x) {
  7. mx[x]=max(mx[x], g);
  8. }
  9. }
  10. int getmx(int x) {
  11. int y=0;
  12. for(; x; x-=x&-x) {
  13. y=max(y, mx[x]);
  14. }
  15. return y;
  16. }
  17. void isort(int *y, int *x) {
  18. for(int i=0; i<=n; ++i) c[i]=0;
  19. for(int i=1; i<=n; ++i) c[y[i]]++;
  20. for(int i=1; i<=n; ++i) c[i]+=c[i-1];
  21. for(int i=1; i<=n; ++i) x[c[y[i]]--]=i;
  22. }
  23. int main() {
  24. scanf("%d\n", &n);
  25. for(int i=1; i<=n; ++i) {
  26. a[i]=getchar()=='p'?1:-1;
  27. }
  28. for(int i=1, j=n; i<=n; ++i, --j) {
  29. s1[i]=s1[i-1]+a[i];
  30. s2[j]=s2[j+1]+a[j];
  31. b[s1[i]+N]=b[s1[i]+1+N]=-1;
  32. c[s2[j]+N]=c[s2[j]+1+N]=n+2;
  33. }
  34. for(int i=1, j=n; i<=n; ++i, --j) {
  35. if(a[i]==1) {
  36. l[i]=b[s1[i]+1+N]+2;
  37. }
  38. if(a[j]==1) {
  39. r[j]=c[s2[j]+1+N]-2;
  40. }
  41. b[s1[i]+N]=i;
  42. c[s2[j]+N]=j;
  43. }
  44. isort(l, b);
  45. int nl=1, t, ans=0;
  46. for(int i=1; i<=n; ++i) {
  47. while(nl<=n && l[t=b[nl]]<=i) {
  48. if(a[t]!=-1) {
  49. add(t, t);
  50. }
  51. ++nl;
  52. }
  53. if(a[i]!=-1) {
  54. ans=max(ans, getmx(r[i])-i+1);
  55. }
  56. }
  57. printf("%d\n", ans);
  58. return 0;
  59. }

【BZOJ】3521: [Poi2014]Salad Bar的更多相关文章

  1. 【BZOJ】3524: [Poi2014]Couriers

    [算法]主席树 [题解]例题,记录和,数字出现超过一半就递归查找. 主席树见[算法]数据结构 #include<cstdio> #include<algorithm> #inc ...

  2. 【BZOJ】3835: [Poi2014]Supercomputer

    题意 \(n(1 \le 1000000)\)个点的有根树,\(1\)号点为根,\(q(1 \le 1000000)\)次询问,每次给一个\(k\),每一次可以选择\(k\)个未访问的点,且父亲是访问 ...

  3. 【BZOJ】3832: [Poi2014]Rally

    题意 \(n(2 \le n \le 500000)\)个点\(m(1 \le m \le 1000000)\)条边的有向无环图,找到一个点,使得删掉这个点后剩余图中的最长路径最短. 分析 神题不会做 ...

  4. 【BZOJ】3526: [Poi2014]Card

    题意 \(n(n \le 200000)\)张卡片,正反有两个数\(a[i], b[i]\).\(m(m \le 1000000)\)次操作,每次交换\(c[i].d[i]\)位置上的卡片.每一次操作 ...

  5. 【BZOJ】3523: [Poi2014]Bricks

    题意 \(n(n \le 1000000)\)个物品,颜色分别为\(a[i]\),现在要求排在一排使得相邻两个砖块的颜色不同,且限定第一个砖块和最后一个砖块的颜色,输出一个合法解否则输出-1. 分析 ...

  6. 【BZOJ】3834: [Poi2014]Solar Panels

    http://www.lydsy.com/JudgeOnline/problem.php?id=3834 题意:求$max\{(i,j)\}, smin<=i<=smax, wmin< ...

  7. 【BZOJ】3524 [POI2014] Couriers(主席树)

    题目 传送门:QWQ 传送到洛谷QWQ 分析 把求区间第k大的改一改就ok了. 代码 #include <bits/stdc++.h> using namespace std; ; ], ...

  8. 【BZOJ】3052: [wc2013]糖果公园

    http://www.lydsy.com/JudgeOnline/problem.php?id=3052 题意:n个带颜色的点(m种),q次询问,每次询问x到y的路径上sum{w[次数]*v[颜色]} ...

  9. 【BZOJ】3319: 黑白树

    http://www.lydsy.com/JudgeOnline/problem.php?id=3319 题意:给一棵n节点的树(n<=1e6),m个操作(m<=1e6),每次操作有两种: ...

随机推荐

  1. MyBatis学习总结(一)——MyBatis快速入门

    一.Mybatis介绍 MyBatis是一个支持普通SQL查询,存储过程和高级映射的优秀持久层框架.MyBatis消除了几乎所有的JDBC代码和参数的手工设置以及对结果集的检索封装.MyBatis可以 ...

  2. LoadRunner 函数之 web_custom_request

    Allows you to create a custom HTTP request with any method supported by HTTP. List of Attributes URL ...

  3. 【强烈推荐】利用NAT、Host-Only双虚拟网卡,实现Virtual Box中CentOS6.3联网

    问题背景: 先前都是在Virtual Box中以“网络共享”方式,让里面的Linux虚拟机Host-Only方式联网,参考如下: Virtual Box下配置Host-Only联网方式详解 但最近被公 ...

  4. ReactiveCocoa源码拆分解析(四)

    (整个关于ReactiveCocoa的代码工程可以在https://github.com/qianhongqiang/QHQReactive下载) 上一章节简要的说明了如何实现的热信号.但是像那么写, ...

  5. 用Bitbucket搭建博客初探

    本博客是搭建在GitHub上的静态博客,但是由于GitHub免费账户不能创建私有仓库,导致有些东西不想放在GitHub上. 前两天,在免费资源部落上发现了Bitbucket,它和GitHub类似,也是 ...

  6. 使用http.sys,让delphi 的多层服务飞起来

    核心提示:一直以来,delphi 的网络通讯层都是以indy 为主,虽然indy 的功能非常多,涉及到网络服务的各个方面,但是对于大多数多层服务来说,就是需要一个快速.稳定.高效的传输层.Delphi ...

  7. Python正则表达式详解

    我用双手成就你的梦想 python正则表达式 ^ 匹配开始 $ 匹配行尾 . 匹配出换行符以外的任何单个字符,使用-m选项允许其匹配换行符也是如此 [...] 匹配括号内任何当个字符(也有或的意思) ...

  8. xmind portable

    portable : http://dl2.xmind.cn/xmind-7.5-update1-portable.zip

  9. Android手机截屏方法

    <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"/> <u ...

  10. sqlserver中分区函数 partition by的用法

    partition  by关键字是分析性函数的一部分,它和聚合函数(如group by)不同的地方在于它能返回一个分组中的多条记录,而聚合函数一般只有一条反映统计值的记录, partition  by ...