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> ...
随机推荐
- SCRUM 12.14
由于最近的课业较多,大家平时有些力不从心,对于工作完成得有限. 最近课业压力小了一些,我们决定从今天起,投入精力. 以下为我们的任务分配情况: 人员 任务 高雅智 清除缓存 彭林江 统计活跃用户数量 ...
- [2017BUAA软工助教]剩余个人作业与deadline
软件工程剩余作业与deadline 标签(空格分隔): 软件工程 一.个人阅读作业+总结 对软件工程的学习做一个总结. 阅读下列关于软件开发本质和开发方法的博客/文章,结合自己在个人项目/结对编程/团 ...
- 带状态论文粗读(三)[引用openstate的相关论文阅读]
一 文章名称:FLOWGUARD: Building Robust Firewalls for Software-Defined Networks 发表时间:2014 期刊来源:--- 解决问题: 一 ...
- 用C语言编程自动生成四则运算
#include<stdio.h>#include<stdlib.h>#include <time.h>#define N 30main(){ int a,b,k, ...
- RYU 灭龙战 second day(内容大部分引自网络)
RYU 灭龙战 second day(内容大部分引自网络) 写好的markdown重启忘了保存...再写一次RLG 巨龙的稀有装备-RYU代码结构 RYU控制器代码结构的总结 RYU入门教程 RYU基 ...
- webSocket开发chat application过程
本次使用websocket开发chat的功能已经接近尾声,等到压力测试结束之后就可以上线了.在此记录一下整个开发过程. ---------------------------------------- ...
- 【补】debug
懒得翻别人博客了,之前的按钮不显示名字,应该是文字ui文件名写错了. 现在不存在任何已知bug.
- Linux下利用json-c从一个json数组中提取每一个元素中的部分字段组成一个新json数组
先把代码贴上来,有时间整理一下 首先说一下要实现的功能: 假定现在有一个json格式的字符串,而且他是一个josn中的数组,比如: [ { "id": "NEW20170 ...
- [知乎]关于WindowsXPx64SP2系统的说明
自己简单安装了下发现 winxpsp3x86的系统版本为: 然后windowsXPx64sp2的版本为: 作者:qpi667链接:https://www.zhihu.com/question/29 ...
- [阮一峰]Linux 守护进程的启动方法
"守护进程"(daemon)就是一直在后台运行的进程(daemon). 本文介绍如何将一个 Web 应用,启动为守护进程. 一.问题的由来 Web应用写好后,下一件事就是启动,让它 ...