bzoj千题计划292:bzoj2244: [SDOI2011]拦截导弹
http://www.lydsy.com/JudgeOnline/problem.php?id=2244
每枚导弹成功拦截的概率 = 包含它的最长上升子序列个数/最长上升子序列总个数
pre_len [i] 表示以i结尾的最长不下降子序列的长度
pre_sum[i] 表示对应长度下的方案数
suf_len[i] 表示以i开头的最长不下降子序列长度
suf_sum[i] 表示对应长度下的方案数
若已有了这4个数组
设最长上升子序列长度=mx
那么 如果pre_len[i]+suf_len[i] - 1==mx,那么第i枚导弹在最长上升子序列上
而且在pre_sum[i]*suf_sum[i] 个最长上升子序列上
如何获取这四个数组?
这是一个三维偏序问题,CDQ分治
若i后面可以接j
第一维:i的出现时间<j的出现时间
第二维:i的高度>=j的高度
第三维:i的速度>=j的速度
具体实现:
排序第一维,CDQ过程中排序解决第二维,树状数组解决第三维
对于第三维,要离散化
树状数组查询的是<=查询位置的信息,即用它来求最长不下降子序列更方便
所以在解决以i为开头的最长不上升子序列的时候
将第二维取负,第三维i变成n-i+1
在解决以i为开始的最长不上升子序列的时候
将第一维取负
对CDQ分治更进一步的理解:
之前写CDQ分治做数据结构题的时候,
写的是先解决左边对右边的贡献,在递归两边
但是这道题,要先递归左边,再解决左边对右边的贡献,再递归右边
因为最长不下降子序列 左边更新右边的时候,左边的所有不是等价的
即也需要用左边更新之后的结果来更新右边
而像那种修改查询题,左边的修改对左边查询的影响和对右边查询的影响是等价的
#include<cstdio>
#include<iostream>
#include<algorithm> using namespace std; #define N 50001 #define lowbit(x) ( x&-x ) int n;
int h[N],v[N];
int tot,has[N]; struct node
{
int x,y,z;
int len;
double sum;
}e[N]; int c[N];
double d[N]; int pre_len[N],suf_len[N];
double pre_sum[N],suf_sum[N]; void read(int &x)
{
x=; char c=getchar();
while(!isdigit(c)) c=getchar();
while(isdigit(c)) { x=x*+c-''; c=getchar(); }
} bool cmdy(node p,node q)
{
if(p.y==q.y) return p.z<q.z;
return p.y<q.y;
} bool cmdx(node p,node q)
{
return p.x<q.x;
} void change(int pos,int len,double sum)
{
while(pos<=n)
{
if(len>c[pos])
{
c[pos]=len;
d[pos]=sum;
}
else if(len==c[pos]) d[pos]+=sum;
pos+=lowbit(pos);
}
} int query_len(int pos)
{
int ans=;
while(pos)
{
ans= ans>=c[pos] ? ans : c[pos];
pos-=lowbit(pos);
}
return ans;
} double query_sum(int pos,int val)
{
double ans=;
while(pos)
{
if(c[pos]==val) ans+=d[pos];
pos-=lowbit(pos);
}
return ans;
} void clear(int pos)
{
while(pos<=n)
{
c[pos]=d[pos]=;
pos+=lowbit(pos);
}
} void cdq(int l,int r)
{
if(l==r) return;
int mid=l+r>>;
cdq(l,mid);
sort(e+l,e+mid+,cmdy);
sort(e+mid+,e+r+,cmdy);
int i=l,j=mid+;
int len; double sum;
for(;j<=r;++j)
{
while(i<=mid && e[i].y<=e[j].y)
{
change(e[i].z,e[i].len,e[i].sum);
i++;
}
len=query_len(e[j].z)+;
sum=query_sum(e[j].z,len-);
if(len>e[j].len)
{
e[j].len=len;
e[j].sum=sum;
}
else if(len==e[j].len) e[j].sum+=sum;
}
for(int k=l;k<=mid;++k) clear(e[k].z);
sort(e+mid+,e+r+,cmdx);
cdq(mid+,r);
} void work()
{
sort(has+,has+n+);
tot=unique(has+,has+n+)-has-;
for(int i=;i<=n;++i) v[i]=lower_bound(has+,has+tot+,v[i])-has;
for(int i=;i<=n;++i)
{
e[i].x=i;
e[i].y=-h[i];
e[i].z=tot-v[i]+;
e[i].len=e[i].sum=;
}
cdq(,n);
for(int i=;i<=n;++i) pre_len[e[i].x]=e[i].len,pre_sum[e[i].x]=e[i].sum;
for(int i=;i<=n;++i)
{
e[i].x=-i;
e[i].y=h[i];
e[i].z=v[i];
e[i].len=e[i].sum=;
}
sort(e+,e+n+,cmdx);
cdq(,n);
for(int i=;i<=n;++i) suf_len[-e[i].x]=e[i].len,suf_sum[-e[i].x]=e[i].sum;
} void get_ans()
{
int mx=;
for(int i=;i<=n;++i) mx=max(mx,pre_len[i]);
printf("%d\n",mx);
double cnt=;
for(int i=;i<=n;++i)
if(pre_len[i]==mx) cnt+=pre_sum[i];
for(int i=;i<=n;++i)
if(pre_len[i]+suf_len[i]-==mx) printf("%.5lf ",(double)pre_sum[i]*suf_sum[i]/cnt);
else printf("0 ");
} int main()
{
read(n);
for(int i=;i<=n;++i)
{
read(h[i]); read(v[i]);
has[i]=v[i];
}
work();
get_ans();
}
bzoj千题计划292:bzoj2244: [SDOI2011]拦截导弹的更多相关文章
- [BZOJ2244][SDOI2011]拦截导弹 CDQ分治
2244: [SDOI2011]拦截导弹 Time Limit: 30 Sec Memory Limit: 512 MB Special Judge Description 某国为了防御敌国的导弹 ...
- bzoj千题计划300:bzoj4823: [Cqoi2017]老C的方块
http://www.lydsy.com/JudgeOnline/problem.php?id=4823 讨厌的形状就是四联通图 且左右各连一个方块 那么破坏所有满足条件的四联通就好了 按上图方式染色 ...
- BZOJ2244: [SDOI2011]拦截导弹(CDQ分治,二维LIS,计数)
Description 某国为了防御敌国的导弹袭击,发展出一种导弹拦截系统.但是这种导弹拦截系统有一个缺陷:虽然它的第一发炮弹能够到达任意的高度.并且能够拦截任意速度的导弹,但是以后每一发炮弹都不能高 ...
- BZOJ2244 [SDOI2011]拦截导弹 【cdq分治 + 树状数组】
题目 某国为了防御敌国的导弹袭击,发展出一种导弹拦截系统.但是这种导弹拦截系统有一个缺陷:虽然它的第一发炮弹能够到达任意的高度.并且能够拦截任意速度的导弹,但是以后每一发炮弹都不能高于前一发的高度,其 ...
- bzoj千题计划254:bzoj2286: [Sdoi2011]消耗战
http://www.lydsy.com/JudgeOnline/problem.php?id=2286 虚树上树形DP #include<cmath> #include<cstdi ...
- bzoj千题计划246:bzoj2242: [SDOI2011]计算器
http://www.lydsy.com/JudgeOnline/problem.php?id=2242 #include<map> #include<cmath> #incl ...
- bzoj千题计划196:bzoj4826: [Hnoi2017]影魔
http://www.lydsy.com/JudgeOnline/problem.php?id=4826 吐槽一下bzoj这道题的排版是真丑... 我还是粘洛谷的题面吧... 提供p1的攻击力:i,j ...
- bzoj千题计划280:bzoj4592: [Shoi2015]脑洞治疗仪
http://www.lydsy.com/JudgeOnline/problem.php?id=4592 注意操作1 先挖再补,就是补的范围可以包含挖的范围 SHOI2015 的题 略水啊(逃) #i ...
- bzoj千题计划177:bzoj1858: [Scoi2010]序列操作
http://www.lydsy.com/JudgeOnline/problem.php?id=1858 2018 自己写的第1题,一遍过 ^_^ 元旦快乐 #include<cstdio> ...
随机推荐
- win10+anaconda3+python3.6+opencv3.1.0
最近用windows系统比较多,就想在win10下搞一下深度学习这一方面的研究,那么就需要配置好环境巴拉巴拉的一堆东西.默默记个笔记,正所谓“好记性不如烂笔头”. 1.安装Anaconda 这个是一个 ...
- Anibei前端基础学习
html.html5.CSS2.CSS3.JQuery.Vue.js学习,后端程序媛开始学习前端开发啦.
- Flink standalone模式作业执行流程
宏观流程如下图: client端 生成StreamGraph env.addSource(new SocketTextStreamFunction(...)) .flatMap(new FlatMap ...
- freopen stdout 真的更快?
freopen stdout 真的更快? 在一次数独作业中,我发现大部分同学提交的代码中都使用 freopen 来将 stdout 重新指向目标文件进行文件输出操作.我感到十分好奇,关于 freope ...
- 团队作业8——测试与发布(Beta阶段)目录
团队作业8——测试与发布(Beta阶段) http://www.cnblogs.com/zy-96/p/8053097.html 团队作业8——测试与发布(Beta阶段)之展示博客 http://ww ...
- 解决局域网IP冲突
进入cmd ipconfig -all 查看现有IP,发现IP不是192.168.1.*的形式,而是192.168.0.*等异常 ipconfig -release 释放现有IP ipconfig ...
- offices 激活
http://www.xitongcheng.com/jiaocheng/dnrj_article_44577.html 破解工具见cnblos文件中 : https://blog.csdn.net ...
- Linux 备份 文件夹的权限 然后在其他机器进行恢复
Study From https://www.cnblogs.com/chenshoubiao/p/4780987.html 用到的命令 getfacl 和 setfacl 备份 getfacl -R ...
- python进阶:Python进程、线程、队列、生产者/消费者模式、协程
一.进程和线程的基本理解 1.进程 程序是由指令和数据组成的,编译为二进制格式后在硬盘存储,程序启动的过程是将二进制数据加载进内存,这个启动了的程序就称作进程(可简单理解为进行中的程序).例如打开一个 ...
- Mybatis中jdbcType和javaType、typeHandler的对照关系
JdbcType与Oracle.MySql数据类型对应列表,及 JdbcType Oracle MySql CHAR CHAR CHAR VARCHAR VARCHAR VARCHAR LONGV ...