设切割的区间为(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. [分享]前端javascript插件(均开源)

    记录并分享一些自己使用过的插件,便于以后有相应的功能需要使用可以及时想到. 1:数字插件countUp.js 功能:用于动态显示数字的增加和减少 项目github地址:http://inorganik ...

  2. 【XSY2968】线性代数

    题目来源:noi2018模拟测试赛(二十二) 毒瘤板题+提答场……真tm爽 提答求最大团,各路神仙退火神仙随机化八仙过海 题意: 题解: 支持双端插入的回文自动机板题 代码: #include< ...

  3. Codeforces 679A Bear and Prime 100

    链接:传送门 题意:给你一个隐藏数,这个隐藏数在[2,100]之间,现在最多可以询问20次,每次询问的是这个数是不是隐藏数的底数,是为yes,不是为no,每次询问后都需要flush一下输出缓冲区,最后 ...

  4. Java Web学习总结(26)——Servlet不同版本之间的区别

    1.   2.3版本 2.3版本 <!DOCTYPE web-app PUBLIC "-//Sun Microsystems, Inc.//DTD Web Application 2. ...

  5. ASP.NET-model验证

    在ASP.NET的model中,可以定义下面的这种属性,来实现前台签证字符串 RegularExpression(@"(|.*(?=.{6,})(?=.*\d)(?=.*[a-zA-Z]). ...

  6. JAVA集合类型(二)

    JAVA集合类型 (现代的变量集群) watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQv/font/5a6L5L2T/fontsize/400/fill/I0J ...

  7. Mysql数据库事务的隔离级别和锁的实现原理分析

    Mysql数据库事务的隔离级别和锁的实现原理分析 找到大神了:http://blog.csdn.net/tangkund3218/article/details/51753243 InnoDB使用MV ...

  8. 110_leetcode_Best Time to Buy and sell Stock II

    Say you have an array for which the ith element is the price of a given stock on day i. Design an al ...

  9. intellij idea 13&amp;14 插件推荐及高速上手建议 (已更新!)

    早些年 在外企的时候,公司用的是intellij idea ,当时也是从eclipse.MyEclipse转过去的非常是不习惯. 用了一周明显感觉爱上它了.由于它非常智能,并且能纠正你非常多不好的习惯 ...

  10. Chisel实验笔记(四)

    在<Chisel实验笔记(二)>中.通过编写TestBench文件,然后使用Icarus Verilog.GtkWave能够測试,查看相关波形.比較直观,在<Chisel实验笔记(三 ...