4384: [POI2015]Trzy wieże

Time Limit: 20 Sec  Memory Limit: 128 MB
Submit: 217  Solved: 61
[Submit][Status][Discuss]

Description

给定一个长度为n的仅包含'B'、'C'、'S'三种字符的字符串,请找到最长的一段连续子串,使得这一段要么只有一种字符,要么有多种字符,但是没有任意两种字符出现次数相同。

Input

第一行包含一个正整数n(1<=n<=1000000),表示字符串的长度。
第二行一个长度为n的字符串。

Output

包含一行一个正整数,即最长的满足条件的子串的长度。

Sample Input

9
CBBSSBCSC

Sample Output

6

HINT

选择BSSBCS这个子串。

Source

[Submit][Status][Discuss]

分析

OTZ Claris

OTZ LincHpin

OTZ MirrorGray

——下面开始正文——

子串只包含一种字符的情况特殊处理,一遍扫描就可以了。

首先将字符串转换称数字串,分别用0,1,2替代B,C,S。

设sum[i][j]表示前i(0<=i<=n)个数字中j(0<=j<=2)的出现次数,则题目要求可以翻译如下

选取一段[lt + 1, rt],满足:

sum[rt][0] - sum[lt][0] != sum[rt][1] - sum[lt][1]

sum[rt][0] - sum[lt][0] != sum[rt][2] - sum[lt][2]

sum[rt][1] - sum[lt][1] != sum[rt][2] - sum[lt][2]

从Claris大神那里学到的机智转换——通过交换等式两边的元素,将lt和rt分离。

sum[rt][0] - sum[rt][1] != sum[lt][0] - sum[lt][1]

sum[rt][0] - sum[rt][2] != sum[lt][0] - sum[lt][2]

sum[rt][1] - sum[rt][2] != sum[lt][1] - sum[lt][2]

这看起来就神清气爽多了,问题就转换成了——

每个点上有一个三元组(x,y,z),以及一个id,求对于每个点来说,和它的三元组完全不同的id相距最远的点。

将(x,y,z,id)按照x从小到大排序,这样就可以保证x的不同了。

然后把y当作树状数组下标维护id的最大、最小值,这样查询的时候避开自己y的地方就可以保证y也不同了。

可以z还可能相同,所以要维护次优解,且强制最优解和次优解的z不同。当最优解的z和当前点的z相同时,选择用次优解来更新即可。这样就保证z也不同了。

然而并不清楚Claris大神是怎么用64行写完的,但也不代表我写200+行就有多麻烦,至少不是很费脑子(才怪)。

代码

 #include <bits/stdc++.h>

 const int N = ;
const int inf = 1e9 + ; int n, m; int h[N]; char s[N]; int sum[]; int answer; struct Point
{
int id, x, y, z; Point(void) {};
Point(int i, int a, int b, int c)
{
id = i;
x = a;
y = b;
z = c;
} friend bool operator < (const Point &a, const Point &b)
{
return a.x < b.x;
}
}p[N]; struct Pair
{
int val, pos; Pair(void) {}; Pair(int a, int b)
{
val = a;
pos = b;
} friend bool operator ^ (const Pair &a, const Pair &b)
{
return a.pos != b.pos;
} friend bool operator < (const Pair &a, const Pair &b)
{
return a.val < b.val;
} friend bool operator > (const Pair &a, const Pair &b)
{
return a.val > b.val;
}
}; struct MinMax
{
Pair min0;
Pair min1;
Pair max0;
Pair max1; MinMax(void)
{
min0 = Pair(inf, -inf);
min1 = Pair(inf, -inf);
max0 = Pair(-inf, -inf);
max1 = Pair(-inf, -inf);
} MinMax(int val, int pos)
{
min0 = max0 = Pair(val, pos);
min1 = Pair(inf, -inf);
max1 = Pair(-inf, -inf);
} friend MinMax operator + (const MinMax &a, const MinMax &b)
{
MinMax r; if (a.min0 < b.min0)
{
r.min0 = a.min0;
if (a.min1 < b.min0)
r.min1 = a.min1;
else if (a.min0 ^ b.min0)
r.min1 = b.min0;
else if (a.min1 < b.min1)
r.min1 = a.min1;
else
r.min1 = b.min1;
}
else
{
r.min0 = b.min0;
if (b.min1 < a.min0)
r.min1 = b.min1;
else if (a.min0 ^ b.min0)
r.min1 = a.min0;
else if (b.min1 < a.min1)
r.min1 = b.min1;
else
r.min1 = a.min1;
} if (a.max0 > b.max0)
{
r.max0 = a.max0;
if (a.max1 > b.max0)
r.max1 = a.max1;
else if (a.max0 ^ b.max0)
r.max1 = b.max0;
else if (a.max1 > b.max1)
r.max1 = a.max1;
else
r.max1 = b.max1;
}
else
{
r.max0 = b.max0;
if (b.max1 > a.max0)
r.max1 = b.max1;
else if (a.max0 ^ b.max0)
r.max1 = a.max0;
else if (b.max1 > a.max1)
r.max1 = b.max1;
else
r.max1 = a.max1;
} return r;
}
}; struct BIT
{
MinMax tree[N]; int lowbit(int x)
{
return x & -x;
} MinMax query(int p)
{
MinMax r; while (p)
{
r = r + tree[p];
p -= lowbit(p);
} return r;
} void insert(int p, MinMax v)
{
while (p <= m)
{
tree[p] = tree[p] + v;
p += lowbit(p);
}
}
}lt, rt; signed main(void)
{
scanf("%d%s", &n, s + ); for (int i = , j = ; i <= n; ++i)
{
if (s[i] == s[i - ])
answer = std::max(answer, ++j);
else
answer = std::max(answer, j = );
} for (int i = ; i <= n; ++i)
{
switch (s[i])
{
case 'B': ++sum[]; break;
case 'C': ++sum[]; break;
case 'S': ++sum[]; break;
} p[i] = Point(
i,
sum[] - sum[],
sum[] - sum[],
sum[] - sum[]
); h[i] = p[i].y;
} p[++n] = Point(, , , ); std::sort(p + , p + + n);
std::sort(h + , h + + n); m = std::unique(h + , h + + n) - h - ; for (int i = ; i <= n; ++i)
p[i].y = std::lower_bound(h + , h + + m, p[i].y) - h; for (int i = , j = ; i <= n; i = j)
{
while (j <= n && p[i].x == p[j].x)++j; for (int k = i; k < j; ++k)
{
MinMax l = lt.query(p[k].y - );
MinMax r = rt.query(m - p[k].y); if (l.min0.pos != p[k].z)
answer = std::max(answer, p[k].id - l.min0.val);
else
answer = std::max(answer, p[k].id - l.min1.val); if (l.max0.pos != p[k].z)
answer = std::max(answer, l.max0.val - p[k].id);
else
answer = std::max(answer, l.max1.val - p[k].id); if (r.min0.pos != p[k].z)
answer = std::max(answer, p[k].id - r.min0.val);
else
answer = std::max(answer, p[k].id - r.min1.val); if (r.max0.pos != p[k].z)
answer = std::max(answer, r.max0.val - p[k].id);
else
answer = std::max(answer, r.max1.val - p[k].id);
} for (int k = i; k < j; ++k)
{
lt.insert(p[k].y, MinMax(p[k].id, p[k].z));
rt.insert(m + - p[k].y, MinMax(p[k].id, p[k].z));
}
} printf("%d\n", answer);
}

BZOJ_4384.cpp

@Author: YouSiki

BZOJ 4384: [POI2015]Trzy wieże的更多相关文章

  1. [POI2015]Trzy wieże

    [POI2015]Trzy wieże 题目大意: 给定一个长度为\(n(n\le10^6)\)的仅包含'B'.'C'.'S'三种字符的字符串,请找到最长的一段连续子串,使得在这一段内出现过的所有字符 ...

  2. 【BZOJ4384】[POI2015]Trzy wieże 树状数组

    [BZOJ4384][POI2015]Trzy wieże Description 给定一个长度为n的仅包含'B'.'C'.'S'三种字符的字符串,请找到最长的一段连续子串,使得这一段要么只有一种字符 ...

  3. BZOJ4384 : [POI2015]Trzy wieże

    首先只有一种字符的情况可以通过双指针在$O(n)$的时间内处理完毕. 设$cnt[i][j]$表示前$i$个字符中$j$字符出现的次数,那么对于两个位置$j<i$: 如果 $cnt[i][0]- ...

  4. bzoj 4386: [POI2015]Wycieczki

    bzoj 4386: [POI2015]Wycieczki 这题什么素质,爆long long就算了,连int128都爆……最后还是用long double卡过的……而且可能是我本身自带大常数吧,T了 ...

  5. BZOJ 4385: [POI2015]Wilcze doły

    4385: [POI2015]Wilcze doły Time Limit: 10 Sec  Memory Limit: 128 MBSubmit: 648  Solved: 263[Submit][ ...

  6. Bzoj 3747: [POI2015]Kinoman 线段树

    3747: [POI2015]Kinoman Time Limit: 60 Sec  Memory Limit: 128 MBSubmit: 553  Solved: 222[Submit][Stat ...

  7. BZOJ 3747 POI2015 Kinoman 段树

    标题效果:有m点,每个点都有一个权值.现在我们有这个m为点的长度n该序列,寻求区间,它仅出现一次在正确的点区间内值和最大 想了很久,甚至神标题,奔说是水的问题--我醉了 枚举左点 对于每个请求留点右键 ...

  8. BZOJ 4380 [POI2015]Myjnie | DP

    链接 BZOJ 4380 题面 有n家洗车店从左往右排成一排,每家店都有一个正整数价格p[i]. 有m个人要来消费,第i个人会驶过第a[i]个开始一直到第b[i]个洗车店,且会选择这些店中最便宜的一个 ...

  9. BZOJ 3747 POI2015 Kinoman

    因为上午没有准备够题目,结果发现写完这道题没题可写了QAQ 又因为这道题范围是100w,我写了发线段树,以为要T,上午就花了一个小时拼命卡常数 结果下午一交居然过了QAQ 我们考虑枚举L,求最大R使得 ...

随机推荐

  1. jmeter(一)基础介绍

    参考书籍:段念<软件性能测试与案例剖析>——第二版 推荐一本书<零成本实现web性能测试——基于Apache—jmeter>,主要内容是一些关于jmeter的实战使用,想学习的 ...

  2. PhpExcel中文帮助手册|PhpExcel使用方法

    下面是总结的几个使用方法 include 'PHPExcel.php'; include 'PHPExcel/Writer/Excel2007.php'; //或者include 'PHPExcel/ ...

  3. MySQL Index详解

    FROM:http://blog.csdn.net/tianmo2010/article/details/7930482 ①MySQL Index 一.SHOW INDEX会返回以下字段 1.Tabl ...

  4. 【点滴积累,厚积薄发】windows schedule task的最小时间间隔是多少?

    在一些业务系统中,经常会出现一些需要定时运行的任务,业界已有很多优秀的选择,比如windows schedule task,Quartz等等.在以前在新蛋的时候用的是内部封装的JobConsole,非 ...

  5. 折腾了1周把程序从sqlserver迁移到oracle上了,每折腾一次需要耗费1周时间

    主要花费时间的事情: 1:安装配套的服务器,安装操作系统,安装数据库,配置远程访问等等,一般会耗费1天时间,甚至2天时间,若手头安装盘不齐全,需要耗费更多时间. 2:远程传输安装文件.特别是开发环境等 ...

  6. Product of Array Except Self

    Given an array of n integers where n > 1, nums, return an array output such that output[i] is equ ...

  7. UDP的坏处

    众所周知,UDP是一个面向无连接的协议.通信时不可靠的.这就会出现一些问题 (1)数据报丢失 因为是无连接,的所以可以用recvfrom和sendto来接收和发送消息,如果socket是阻塞的,那么当 ...

  8. ubuntu13.04环境hadoop1.2.1单机模式安装

    一.虚拟机上安裝ubuntun 13.04 中文版 当然,你要是习惯看英文版,也可以直接安装英文版.老老实实从官网下载安装即可,安装系统不是本文的重点.这里只提一个注意事项:新手安装前,切记断网,因为 ...

  9. JavaScript文件加载器LABjs API详解

    在<高性能JavaScript>一书中提到了LABjs这个用来加载JavaScript文件的类库,LABjs是Loading And Blocking JavaScript的缩写,顾名思义 ...

  10. WPF循环加载图片导致内存溢出的解决办法

    程序场景:一系列的图片,从第一张到最后一张依次加载图片,形成“动画”. 生成BitmapImage的方法有多种: 1. var source=new BitmapImage(new Uri(" ...