Portal

Description

给出一个\(n(n\leq10^5)\)个数的数列\(\{a_n\}\)和\(m(m\leq10^5)\)个形如\((x,y)\)的变化,表示\(a_x\)可以变成\(y\)。我们称一个子序列是合法的当且仅当其中至多一个数发生变化后,其仍然是一个不降序列。求最长合法子序列的长度。

Solution

记\(a_i\)能变化的最小值为\(fr_i\),最大值为\(to_i\),\(dp[i]\)表示以\(a_i\)结尾的最长合法子序列的长度。则很容易得到:

\[ dp[i]=max\{dp[j]\}+1 \quad (j<i,to_j\leq a_i,a_j\leq fr_i) $$ 时间复杂度为$O(n^2)$。
考虑如何快速求出满足$to_j\leq a_i,a_j\leq fr_i$的$j$,可以用k-d树搞(可能被卡常),这里用CDQ分治解决。
每次考虑$dp[L..mid]$对$dp[mid+1..R]$的影响。对于$i\in[mid+1,R]$记录三元组$(a_i,fr_i,i)$,对于$j\in[L,mid]$记录三元组$(to_j,a_j,j)$,排序后用线段树可搞。具体来说,线段树上区间$[L_0,R_0]$维护满足$a_j\in[L_0,R_0]$的$dp[j]$的最大值。排序保证了$to_j\leq a_i$,并且询问处于最后。
> 时间复杂度$O(nlog^2n)$。

##Code
```cpp
//序列
#include <algorithm>
#include <cstdio>
#include <cstring>
using namespace std;
inline char gc()
{
static char now[1<<16],*s,*t;
if(s==t) {t=(s=now)+fread(now,1,1<<16,stdin); if(s==t) return EOF;}
return *s++;
}
inline int read()
{
int x=0; char ch=gc();
while(ch<'0'||'9'<ch) ch=gc();
while('0'<=ch&&ch<='9') x=x*10+ch-'0',ch=gc();
return x;
}
int const N=1e5+10;
int const V=N;
int n,m;
struct rec{int a,fr,to,id;} r[N],r1[N];
int cmpM;
bool cmpR(rec x,rec y)
{
int x1,x2,y1,y2;
if(x.id<=cmpM) x1=x.to,x2=x.a;
else x1=x.a,x2=x.fr;
if(y.id<=cmpM) y1=y.to,y2=y.a;
else y1=y.a,y2=y.fr;
if(x1==y1) return x2==y2?x.id<y.id:x2<y2;
else return x1<y1;
}
int dp[N];
#define Ls (p<<1)
#define Rs ((p<<1)|1)
int rt; int maxV[N<<2],tag[N<<2];
void update(int p) {maxV[p]=max(maxV[Ls],maxV[Rs]);}
void change(int p,int x) {maxV[p]=tag[p]=x;}
void pushdw(int p) {if(tag[p]!=-1) change(Ls,tag[p]),change(Rs,tag[p]),tag[p]=-1;}
int L,R; int tr[N];
void ins(int p,int L0,int R0,int x)
{
if(L==L0&&R0==L) {change(p,max(maxV[p],x)); return;}
pushdw(p);
int mid=L0+R0>>1;
if(L<=mid) ins(Ls,L0,mid,x);
else ins(Rs,mid+1,R0,x);
update(p);
}
int query(int p,int L0,int R0)
{
if(L<=L0&&R0<=R) return maxV[p];
pushdw(p);
int mid=L0+R0>>1,res=0;
if(L<=mid) res=max(res,query(Ls,L0,mid));
if(mid<R) res=max(res,query(Rs,mid+1,R0));
return res;
}
void solve(int fr,int to)
{
if(fr==to) return;
int mid=fr+to>>1;
solve(fr,mid);
for(int i=fr;i<=to;i++) r1[i]=r[i];
cmpM=mid; sort(r1+fr,r1+to+1,cmpR);
change(rt,0);
for(int i=fr;i<=to;i++)
{
int t=r1[i].id;
if(t<=mid) L=r[t].a,ins(rt,1,V,dp[t]);
else L=1,R=r[t].fr,dp[t]=max(dp[t],query(rt,1,V)+1);
}
solve(mid+1,to);
}
int main()
{
n=read(),m=read();
for(int i=1;i<=n;i++) r[i].fr=r[i].to=r[i].a=read(),r[i].id=i;
for(int i=1;i<=m;i++)
{
int x=read(),y=read();
r[x].fr=min(r[x].fr,y),r[x].to=max(r[x].to,y);
}
rt=1; memset(tag,-1,sizeof tag);
for(int i=1;i<=n;i++) dp[i]=1;
solve(1,n);
int ans=0;
for(int i=1;i<=n;i++) ans=max(ans,dp[i]);
printf("%d\n",ans);
return 0;
}
```\]

BZOJ4553 - [TJOI2016]序列的更多相关文章

  1. 【BZOJ4553】[HAOI2016&TJOI2016]序列

    [BZOJ4553][HAOI2016&TJOI2016]序列 题面 bzoj 洛谷 题解 一定要仔细看题啊qwq... 我们设$mn[i],mx[i]$表示第$i$个位置上最小出现.最大出现 ...

  2. [BZOJ4553][HEOI2016]序列 CDQ分治

    4553: [Tjoi2016&Heoi2016]序列 Time Limit: 20 Sec  Memory Limit: 128 MB Description 佳媛姐姐过生日的时候,她的小伙 ...

  3. cdq分治(hdu 5618 Jam's problem again[陌上花开]、CQOI 2011 动态逆序对、hdu 4742 Pinball Game、hdu 4456 Crowd、[HEOI2016/TJOI2016]序列、[NOI2007]货币兑换 )

    hdu 5618 Jam's problem again #include <bits/stdc++.h> #define MAXN 100010 using namespace std; ...

  4. 洛谷 P4093 [HEOI2016/TJOI2016]序列 解题报告

    P4093 [HEOI2016/TJOI2016]序列 题目描述 佳媛姐姐过生日的时候,她的小伙伴从某宝上买了一个有趣的玩具送给他.玩具上有一个数列,数列中某些项的值可能会变化,但同一个时刻最多只有一 ...

  5. 洛谷 P4093 [HEOI2016/TJOI2016]序列 CDQ分治优化DP

    洛谷 P4093 [HEOI2016/TJOI2016]序列 CDQ分治优化DP 题目描述 佳媛姐姐过生日的时候,她的小伙伴从某宝上买了一个有趣的玩具送给他. 玩具上有一个数列,数列中某些项的值可能会 ...

  6. [BZOJ4553][TJOI2016&&HEOI2016]序列(CDQ分治)

    4553: [Tjoi2016&Heoi2016]序列 Time Limit: 20 Sec  Memory Limit: 128 MBSubmit: 1202  Solved: 554[Su ...

  7. [BZOJ4553][Tjoi2016&Heoi2016]序列 cdp分治+dp

    4553: [Tjoi2016&Heoi2016]序列 Time Limit: 20 Sec  Memory Limit: 128 MBSubmit: 260  Solved: 133[Sub ...

  8. bzoj4553 [Tjoi2016&Heoi2016]序列 树状数组(区间最大值)+cqd

    [Tjoi2016&Heoi2016]序列 Time Limit: 20 Sec  Memory Limit: 128 MBSubmit: 1006  Solved: 464[Submit][ ...

  9. BZOJ4553: [Tjoi2016&Heoi2016]序列

    Description 佳媛姐姐过生日的时候,她的小伙伴从某宝上买了一个有趣的玩具送给他.玩具上有一个数列,数列中某些项的值 可能会变化,但同一个时刻最多只有一个值发生变化.现在佳媛姐姐已经研究出了所 ...

随机推荐

  1. php一致性hash性能测试(flexihash/memcache/memcached)

    一致性hash的使用在PHP中有三种选择分别是原生的memcache扩展,memcached扩展,还有一个是网上比较流行的flexihash类. 最近有项目需要使用flexihash类操作memcac ...

  2. C#基础学习2

    变量与数据类型!

  3. 在Android上使用酷狗歌词API

    参考自http://blog.csdn.net/u010752082/article/details/50810190 代码先贴出来: public void searchLyric(){ final ...

  4. dataTables去掉搜索框和每页多少条框体,解决Cannot reinitialise DataTable问题

    $('#example').DataTable({ searching:false, //去掉搜索框 bLengthChange:false,//去掉每页多少条框体 "language&qu ...

  5. # Transition:添加弹出过渡效果

    # Transition:添加弹出过渡效果 通过鼠标的单击.获得焦点,被点击或对元素任何改变中触发,并平滑地以动画效果改变CSS的属性值. W3C-transition MDN-transition ...

  6. Python 语言规范

    Python 语言规范 pychecker  对你的代码运行pychecker 定义: pychecker 是一个在Python 源代码中查找bug 的工具. 对于C 和C++这样的不那 么动态的( ...

  7. springBoot + KISSO实现单点登录

    1:创建一个maven项目 kisso,然后再创建二个子项目都是springboot 2:二个boot项目的pom.xml都是一样的 就这三个依赖,3:接下来就是码代码了,首先在(在我这里)sprin ...

  8. 中间件及tomcat的内存溢出调优

    主要是这三个选项的调整需要根据主机的内存配置 以及业务量的使用情况调节 -Xmx4g -Xms4g -Xmn2g xmx 与xms一般设置为一样 xmn大致设置为xmx xms的三分之一   可以使用 ...

  9. Python3基础教程(十五)—— PEP8 代码风格指南

    编程语言不是艺术,而是工作或者说是工具,所以整理并遵循一套编码规范是十分必要的. 这篇文章原文实际上来自于这里:https://www.python.org/dev/peps/pep-0008/ 有很 ...

  10. QT +坐标系统 + 自定义控件 + 对象树的验证(自动进行析构)_内存回收机制

    通过创建一个新的按钮类,来进行析构函数的验证,即对象树概念的验证.当程序结束的时候会自动的调用析构函数, 验证思路: 要验证按钮会不会自动的析构,(即在QPushButton类里面的析构函数添加qDe ...