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> ...
随机推荐
- effective c++ 笔记 (35-40)
//---------------------------15/04/24---------------------------- //#35 考虑virtual函数以外的其他选择 { /* 1: ...
- Reflux系列01:异步操作经验小结
写在前面 在实际项目中,应用往往充斥着大量的异步操作,如ajax请求,定时器等.一旦应用涉及异步操作,代码便会变得复杂起来.在flux体系中,让人困惑的往往有几点: 异步操作应该在actions还是s ...
- kubernetes部署mysql
第一章 部署K8S集群 https://www.cnblogs.com/zoulixiang/p/9504324.html 第二章 1.新建mysql-rc.yaml vi mysql-rc.yaml ...
- Keyshot+AD渲染PCB效果图
Keyshot+AD渲染PCB效果图 1.前言 前些天,公司同事找到我说,公司的展会宣传册要更新的了,有几款新的产品需要更新添加上去,大部分的新产品都有实物demo,可以拍照修一下图弄上去.但不巧,其 ...
- 如何自出版一本书:定制 bookdown
目录 如何自出版一本书:定制 bookdown bookdown 的第一步 亚马逊 Kindle 格式 创建书籍 _bookdown.yml 注意行宽 写在每个 .Rmd 文件的开头 index.Rm ...
- 用C语言编程自动生成四则运算
#include<stdio.h>#include<stdlib.h>#include <time.h>#define N 30main(){ int a,b,k, ...
- 新安装的Ubuntu设置root密码
一.问题描述 新安装的Ubuntu切换到root用户时如果没有设置root用户密码会操作失败.此时需要先设置root用户密码. 二.解决办法 打开终端执行 sudo passwd 命令. 输入设置的密 ...
- 【菜鸟】RESTful 架构详解
RESTful 架构详解 分类 编程技术 1. 什么是REST REST全称是Representational State Transfer,中文意思是表述(编者注:通常译为表征)性状态转移. 它首次 ...
- OneZero第一次站立会议&Sprint Planning Meeting(2016.3.21)
会议时间:2016年3月21日 11:40~12:00 会议成员:冉华,张敏,王巍,夏一鸣. 会议目的:列举第一周工作内容,确定第一周工作分配.即挑选出一个Story作为本次迭代完成的目标. 会议内容 ...
- Fitts’ Law / 菲茨定律(费茨法则)
Fitts’ Law / 菲茨定律(费茨法则) 补充一张雅虎ued绘制的关于Fitts’ Law的Q版小漫画,先初步了解下: Fitts’ Law / 菲茨定律(费茨法则) Fitts’ Law ...