设切割的区间为(j, i), 注意两边都是开区间。

然后可以预处理出以i为起点的最长连续递增的长度和以j为终点的最长连续递增的长度。

大致思路就是枚举i,右边这一侧的最优值就知道了, 然后这道题的关键就是就是j取哪里。
(1)去掉干扰元素, 这一步非常的关键, 设题目给的数组为a, g(i)表示以i为结尾的最长递增序列长度
在j < i中, 如果 a[j'] <= a[j] 同时 g(j') > g(j), 那么 j这个元素肯定不是最优的。因为如果j可以取的话

j'就一定可以取, 而且更优。如果j可以取, 就满足a[j] < a[i], 而a[j'] <= a[j], 所以a[j'] < a[i],所以j'可以取
同时g(j') > g(j),代表这个长度更长, 答案更优。所以j这个元素如果可以取得话一定可以被j’代替, 所以
j就对答案没有影响, 就去掉。
这样去掉以后, 就代表如果按照a[j]来排序得到一个序列的话, g(j)就是递增的(不递增的元素就会舍去, 按照之前的结论)
这一步排除这些元素非常关键,因为如果是递增的话, 就可以二分了。
(2) 但是这里有个问题, 因为这个序列是不断要改变的, 也就是说顺序是不断改变的, 但是又不可能每一次
都排一次a[i]来保证单调性, 所以就用到了set, 也就是我标题说的用set来实现动态变化中的二分。因为set本身
就是排好序的, 所以元素增加删除的时候会自动调整, 而set里面还有自带的二分lower_bound, 就非常的方便了。

(4)所以就是每一次都二分去找最优的j, 然后去更新答案和改变set里面的值了。代码中有一句话是删除掉c
为什么呢?设原来的c叫做c0, 现在的c就是c。
第一, 如果这个c0本来就不在set中的话, 那么删除是没事的。第二, 如果这个c0在set中的话,
那么现在加入的c肯定更优, 也就是说原来的c0.g肯定是小于c.g
为什么呢? 假设c0的那个递增序列的c0的前一个位置为p.
如果在set中c0的前一个值就是p的话,那么c.g是等于c0.g的长度的,如果不等于的话keep就等于false了,接下来c就不会插入了
如果set中c0的前一个值是在p和c0之间的话, 那么c.g是大于c0.g的长度的,因为set中是递增的。

这个可能有点抽象, 大家可以拿笔画一下, 我也是理解了很久才想明白的。



#include<cstdio>
#include<set>
#include<algorithm>
#define REP(i, a, b) for(int i = (a); i < (b); i++)
using namespace std; const int MAXN = 212345;
int a[MAXN], f[MAXN], g[MAXN], n;
struct node
{
int a, g;
node(int a, int g) : a(a), g(g) {}
bool operator < (const node& rhs) const
{
return a < rhs.a;
}
};
set<node> s; int main()
{
int T;
scanf("%d", &T);
while(T--)
{
scanf("%d", &n);
REP(i, 0, n) scanf("%d", &a[i]);
if(n == 1) { puts("1"); continue; } g[0] = 1;
REP(i, 1, n)
{
if(a[i] > a[i-1]) g[i] = g[i-1] + 1;
else g[i] = 1;
} f[n-1] = 1;
for(int i = n - 2; i >= 0; i--)
{
if(a[i] < a[i+1]) f[i] = f[i+1] + 1;
else f[i] = 1;
} s.clear();
s.insert(node(a[0], g[0]));
int ans = 1;
REP(i, 1, n)
{
node c(a[i], g[i]);
set<node>::iterator it = s.lower_bound(c);
bool keep = true; if(it != s.begin())
{
node last = *(--it);
ans = max(ans, f[i] + last.g);
if(last.g >= c.g) keep = false;
} if(keep)
{
s.erase(c);
s.insert(c);
it = s.find(c);
it++;
while(it != s.end() && it->a > c.a && it->g <= c.g) s.erase(it++);
}
} printf("%d\n", ans);
} return 0;
}

紫书 例题8-8 UVa 1471 (用set实现动态二分)的更多相关文章

  1. 紫书 例题 11-13 UVa 10735(混合图的欧拉回路)(最大流)

    这道题写了两个多小时-- 首先讲一下怎么建模 我们的目的是让所有点的出度等于入度 那么我们可以把点分为两部分, 一部分出度大于入度, 一部分入度大于出度 那么显然, 按照书里的思路,将边方向后,就相当 ...

  2. 紫书 例题8-3 UVa 1152(中途相遇法)

    这道题要逆向思维, 就是求出答案的一部分, 然后反过去去寻找答案存不存在. 其实很多其他题都用了这道题目的方法, 自己以前都没有发现, 这道题专门考这个方法.这个方法可以没有一直往下求, 可以省去很多 ...

  3. 紫书 例题8-12 UVa 12627 (找规律 + 递归)

    紫书上有很明显的笔误, 公式写错了.g(k, i)的那个公式应该加上c(k-1)而不是c(k).如果加上c(k-1)那就是这一次 所有的红气球的数目, 肯定大于最下面i行的红气球数 我用的是f的公式, ...

  4. 紫书 例题8-4 UVa 11134(问题分解 + 贪心)

     这道题目可以把问题分解, 因为x坐标和y坐标的答案之间没有联系, 所以可以单独求两个坐标的答案 我一开始想的是按照左区间从小到大, 相同的时候从右区间从小到大排序, 然后WA 去uDebug找了数据 ...

  5. 紫书 例题8-17 UVa 1609 (构造法)(详细注释)

    这道题用构造法, 就是自己依据题目想出一种可以得到解的方法, 没有什么规律可言, 只能根据题目本身来思考. 这道题的构造法比较复杂, 不知道刘汝佳是怎么想出来的, 我想的话肯定想不到. 具体思路紫书上 ...

  6. 紫书 例题 9-5 UVa 12563 ( 01背包变形)

    总的来说就是价值为1,时间因物品而变,同时注意要刚好取到的01背包 (1)时间方面.按照题意,每首歌的时间最多为t + w - 1,这里要注意. 同时记得最后要加入时间为678的一首歌曲 (2)这里因 ...

  7. 紫书 例题 10-2 UVa 12169 (暴力枚举)

    就是暴力枚举a, b然后和题目给的数据比较就ok了. 刘汝佳这道题的讲解有点迷,书上讲有x1和a可以算出x2, 但是很明显x2 = (a * x1 +b) 没有b怎么算x2?然后我就思考了很久,最后去 ...

  8. 紫书 例题 10-26 UVa 11440(欧拉函数+数论)

    这里用到了一些数论知识 首先素因子都大于M等价与M! 互质 然后又因为当k与M!互质且k>M!时当且仅当k mod M! 与M!互质(欧几里得算法的原理) 又因为N>=M, 所以N!为M! ...

  9. 紫书 例题8-2 UVa 11605(构造法)

    这道题方法非常的巧妙, 两层的n*n, 第一层第I行全是第I个国家, 第二层的第j列全是第j个国家.这样能符合题目的条件.比如说第1个国家, 在第一层的第一行全是A, 然后在第二层的第一行就有ABCD ...

随机推荐

  1. NOI 2018 你的名字 (后缀自动机+线段树合并)

    题目大意:略 令$ION2017=S,ION2018=T$ 对$S$建$SAM$,每次都把$T$放进去跑,求出结尾是i的前缀串,能匹配上$S$的最长后缀长度为$f_{i}$ 由于$T$必须在$[l,r ...

  2. PHP下的异步尝试一:初识生成器

    PHP下的异步尝试系列 PHP下的异步尝试一:初识生成器 PHP下的异步尝试二:初识协程 PHP下的异步尝试三:协程的PHP版thunkify自动执行器 PHP下的异步尝试四:PHP版的Promise ...

  3. myeclipse如何取消某一个文件的校验

  4. MD5加密技术

    前几天,在看OpenVXI3.4的时候,偶然发现了几个奇怪的文件,那就是OpenVXI-3.4\src\cache下面的,base64.c,base64.h,md5.c,md5.h.既然有人把源代码给 ...

  5. 2015 Multi-University Training Contest 7 hdu 5373 The shortest problem

    The shortest problem Time Limit: 3000/1500 MS (Java/Others)    Memory Limit: 65536/65536 K (Java/Oth ...

  6. 洛谷 P1301 魔鬼之城

    P1301 魔鬼之城 题目描述 在一个被分割为N*M个正方形房间的矩形魔鬼之城中,一个探险者必须遵循下列规则才能跳跃行动.他必须从(1, 1)进入,从(N, M)走出:在每一房间的墙壁上都写了一个魔法 ...

  7. NHibernate3剖析:Mapping篇之集合映射基础(3):List映射

    系列引入 NHibernate3.0剖析系列分别从Configuration篇.Mapping篇.Query篇.Session策略篇.应用篇等方面全面揭示NHibernate3.0新特性和应用及其各种 ...

  8. hdu5389(DP)

    题意: 给出n个人的id,有两个门,每一个门有一个标号.我们记作a和b,如今我们要将n个人分成两组,进入两个门中,使得两部分人的标号的和(迭代的求,直至变成一位数.我们姑且叫做求"和&quo ...

  9. 推断CPU 是小端存储(Little endian)还是大端存储(Big endian)模式

    第一个版本号: //return true in big-endian machines bool check_big_endian1() { int a = 0; int *p = &a; ...

  10. libLAS1.8.0 编译和配置(VS2013+Win7 64)(一)

    libLAS 是一个用来读写三维激光雷达数据(LiDAR) 的 C++ 库.在学习.科研和研发中都会广泛运用.怎样编译和配置自己所须要版本号的libLAS库确是一件麻烦耗时的事情. 笔者在Win7 6 ...