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

题目描述

佳媛姐姐过生日的时候,她的小伙伴从某宝上买了一个有趣的玩具送给他。

玩具上有一个数列,数列中某些项的值可能会变化,但同一个时刻最多只有一个值发生变化。现在佳媛姐姐已经研究出了所有变化的可能性,她想请教你,能否选出一个子序列,使得在任意一种变化中,这个子序列都是不降的?请你告诉她这个子序列的最长长度即可。

输入格式

输入的第一行有两个正整数 \(n,m\),分别表示序列的长度和变化的个数。

接下来一行有 \(n\) 个整数,表示这个数列原始的状态。

接下来 \(m\) 行,每行有 \(2\) 个整数 \(x,y\),表示数列的第 \(x\) 项可以变化成 \(y\) 这个值。

输出格式

输出一个整数,表示对应的答案。

输入输出样例

输入 #1

3 4

1 2 3

1 2

2 3

2 1

3 4

输出 #1

3

说明/提示

注意:每种变化最多只有一个值发生变化。

在样例输入中,所有的变化是:

1 2 3

2 2 3

1 3 3

1 1 3

1 2 4

选择子序列为原序列,即在任意一种变化中均为不降子序列。

对于 \(20\%\) 数据,所有数均为正整数,且小于等于 \(300\)。

对于 \(50\%\) 数据,所有数字均为正整数,且小于等于 \(3000\)。

对于 \(100\%\) 数据,所有数字均为正整数,且小于等于 \(10^5\)。\(1\le x\le n\)。

分析

我们设\(min[i]\)为处在位置\(i\)上的数变化得到的最小值,\(max[i]\)为处在位置\(i\)上的数变化得到的最大值,\(f[i]\)为以\(i\)结尾的最长上升子序列的长度

则\(f[i]=max(f[i],f[j]+1),j<i,max[j] \leq i,min[i] \geq j\)\

我们会发现这是一个三位偏序问题,可以用\(CDQ\)分治优化

代码

#include<cstdio>
#include<algorithm>
#include<iostream>
inline int read(){
int x=0,fh=1;
char ch=getchar();
while(ch<'0' || ch>'9'){
if(ch=='-') fh=-1;
ch=getchar();
}
while(ch>='0' && ch<='9'){
x=(x<<1)+(x<<3)+(ch^48);
ch=getchar();
}
return x*fh;
}
const int maxn=1e6+5;
int f[maxn],a[maxn],mmax[maxn],mmin[maxn],tr[maxn],n,m;
int lb(int xx){
return xx&-xx;
}
void ad(int wz,int val){
for(int i=wz;i<maxn;i+=lb(i)){
tr[i]=std::max(tr[i],val);
}
}
int cx(int wz){
int ans=0;
for(int i=wz;i>0;i-=lb(i)){
ans=std::max(tr[i],ans);
}
return ans;
}
void qk(int wz){
for(int i=wz;i<maxn;i+=lb(i)){
tr[i]=0;
}
}
int tot=1,id[maxn];
bool cmp1(int aa,int bb){
return a[aa]<a[bb];
}
bool cmp2(int aa,int bb){
return mmin[aa]<mmin[bb];
}
void solve(int l,int r){
if(l==r) return;
int mids=(l+r)>>1;
solve(l,mids);
for(int i=l;i<=r;i++) id[i]=i;
std::sort(id+l,id+mids+1,cmp1);
std::sort(id+mids+1,id+r+1,cmp2);
int now=l;
for(int i=mids+1;i<=r;i++){
while(a[id[now]]<=mmin[id[i]] && now<=mids){
ad(mmax[id[now]],f[id[now]]);
now++;
}
f[id[i]]=std::max(f[id[i]],cx(a[id[i]])+1);
tot=std::max(tot,f[id[i]]);
}
for(int i=now-1;i>=l;i--){
qk(mmax[id[i]]);
}
solve(mids+1,r);
}
int main(){
n=read(),m=read();
for(int i=1;i<=n;i++){
a[i]=read();
mmax[i]=mmin[i]=a[i];
f[i]=1;
}
for(int i=1;i<=m;i++){
int aa,bb;
aa=read(),bb=read();
mmax[aa]=std::max(mmax[aa],bb);
mmin[aa]=std::min(mmin[aa],bb);
}
solve(1,n);
printf("%d\n",tot);
return 0;
}

洛谷 P4093 [HEOI2016/TJOI2016]序列 CDQ分治优化DP的更多相关文章

  1. BZOJ4553/洛谷P4093 [HEOI2016/TJOI2016]序列 动态规划 分治

    原文链接http://www.cnblogs.com/zhouzhendong/p/8672434.html 题目传送门 - BZOJ4553 题目传送门 - 洛谷P4093 题解 设$Li$表示第$ ...

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

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

  3. 洛谷 P4093 [HEOI2016/TJOI2016]序列(Cdq+dp)

    题面 luogu 题解 \(Cdq分治+dp\) \(mx[i],mn[i]\)分别表示第\(i\)位最大,最小能取到多少 那么有 \(j < i\) \(mx[j] \le a[i]\) \( ...

  4. 洛谷P4093 [HEOI2016/TJOI2016]序列

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

  5. 【洛谷P4093】 [HEOI2016/TJOI2016]序列 CDQ分治+动态规划

    你发现只会改变一个位置,所以可以直接进行dp 具体转移的话用 CDQ 分治转移就好了~ #include <bits/stdc++.h> #define N 100006 #define ...

  6. 洛谷 P5331 - [SNOI2019]通信(CDQ 分治优化建图+费用流)

    题面传送门 首先熟悉网络流的同学应该能一眼看出此题的建模方法: 将每个点拆成两个点 \(in_i,out_i\),连一条 \(S\to in_i\),容量为 \(1\) 费用为 \(0\) 的边 连一 ...

  7. [HEOI2016/TJOI2016]序列 CDQ分治

    ---题面--- 题解: 首先我们观察一下,如果一个点对(j, i), 要符合题中要求要满足哪些条件? 首先我们设 j < i 那么有: j < i max[j] < v[i] v[ ...

  8. luogu4093 序列 (cdq分治优化dp)

    设f[i]是以i位置为结尾的最长满足条件子序列的长度 那么j能转移到i的条件是,$j<i , max[j]<=a[i] , a[j]<=min[i]$,其中max和min表示这个位置 ...

  9. 洛谷 P3580 - [POI2014]ZAL-Freight(单调队列优化 dp)

    洛谷题面传送门 考虑一个平凡的 DP:我们设 \(dp_i\) 表示前 \(i\) 辆车一来一回所需的最小时间. 注意到我们每次肯定会让某一段连续的火车一趟过去又一趟回来,故转移可以枚举上一段结束位置 ...

随机推荐

  1. 分享:C语言大礼包(PDF)

    链接:https://pan.baidu.com/s/1xpOcT1gRhdumV5MlkC4AFQ 提取码:4kh6

  2. docker找回构建时被删除的文件

    设想这样一个场景:当一个docker镜像被多次引用构建,在某次构建中某个文件被删除,如何找回被删除的文件? 要想回答这么一个问题,首先得熟悉下docker镜像的分层存储结构,镜像每一层都是只读的: 那 ...

  3. artifactdescriptorexception:Failed to read artifact descriptor for xxx:jar ”

    在Eclipse中执行Maven的install命令时,报“Failed to read artifact descriptor for xxx:jar ”的错误.这可能是在下载过程中文件出现错误.或 ...

  4. C#LeetCode刷题之#709-转换成小写字母(To Lower Case)

    问题 该文章的最新版本已迁移至个人博客[比特飞],单击链接 https://www.byteflying.com/archives/3965 访问. 实现函数 ToLowerCase(),该函数接收一 ...

  5. golang的fmt

    前言 不做文字搬运工,多做思路整理 就是为了能速览标准库,只整理我自己看过的...... 注意!!!!!!!!!! 单词都是连着的,我是为了看着方便.理解方便才分开的 1.fmt 中文文档 [英文文档 ...

  6. python中操作csv文件

    python中操作csv文件 读取csv improt csv f = csv.reader(open("文件路径","r")) for i in f: pri ...

  7. 45道Promise面试题

    来看看通过阅读本篇文章要点: Promise的几道基础题 Promise结合setTimeout Promise中的then.catch.finally Promise中的all和race async ...

  8. 谈谈javascript的基本规范~~~~

    1.不要在同一行声明多个变量. 2.请使用===或==来比较true或false或者数值 3.使用对象字面量代替new Array这种形式 4.不要使用全局函数 5.switch语句必须带有defau ...

  9. SparkStreaming架构

    SparkStreaming是一个对实时数据流进行高通量.容错处理的流式处理系统,可以对多种数据源(如Kdfka.Flume.Twitter.Zero和TCP 套接字)进行类似Map.Reduce和J ...

  10. linux zip压缩文件忽略指定的文件夹

    zip -r productImages.zip ./* -x "cache**" 压缩时,会忽略cache下的所有文件及文件夹