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]拦截导弹的更多相关文章

  1. [BZOJ2244][SDOI2011]拦截导弹 CDQ分治

    2244: [SDOI2011]拦截导弹 Time Limit: 30 Sec  Memory Limit: 512 MB  Special Judge Description 某国为了防御敌国的导弹 ...

  2. bzoj千题计划300:bzoj4823: [Cqoi2017]老C的方块

    http://www.lydsy.com/JudgeOnline/problem.php?id=4823 讨厌的形状就是四联通图 且左右各连一个方块 那么破坏所有满足条件的四联通就好了 按上图方式染色 ...

  3. BZOJ2244: [SDOI2011]拦截导弹(CDQ分治,二维LIS,计数)

    Description 某国为了防御敌国的导弹袭击,发展出一种导弹拦截系统.但是这种导弹拦截系统有一个缺陷:虽然它的第一发炮弹能够到达任意的高度.并且能够拦截任意速度的导弹,但是以后每一发炮弹都不能高 ...

  4. BZOJ2244 [SDOI2011]拦截导弹 【cdq分治 + 树状数组】

    题目 某国为了防御敌国的导弹袭击,发展出一种导弹拦截系统.但是这种导弹拦截系统有一个缺陷:虽然它的第一发炮弹能够到达任意的高度.并且能够拦截任意速度的导弹,但是以后每一发炮弹都不能高于前一发的高度,其 ...

  5. bzoj千题计划254:bzoj2286: [Sdoi2011]消耗战

    http://www.lydsy.com/JudgeOnline/problem.php?id=2286 虚树上树形DP #include<cmath> #include<cstdi ...

  6. bzoj千题计划246:bzoj2242: [SDOI2011]计算器

    http://www.lydsy.com/JudgeOnline/problem.php?id=2242 #include<map> #include<cmath> #incl ...

  7. bzoj千题计划196:bzoj4826: [Hnoi2017]影魔

    http://www.lydsy.com/JudgeOnline/problem.php?id=4826 吐槽一下bzoj这道题的排版是真丑... 我还是粘洛谷的题面吧... 提供p1的攻击力:i,j ...

  8. bzoj千题计划280:bzoj4592: [Shoi2015]脑洞治疗仪

    http://www.lydsy.com/JudgeOnline/problem.php?id=4592 注意操作1 先挖再补,就是补的范围可以包含挖的范围 SHOI2015 的题 略水啊(逃) #i ...

  9. bzoj千题计划177:bzoj1858: [Scoi2010]序列操作

    http://www.lydsy.com/JudgeOnline/problem.php?id=1858 2018 自己写的第1题,一遍过 ^_^ 元旦快乐 #include<cstdio> ...

随机推荐

  1. effective c++ 笔记 (35-40)

    //---------------------------15/04/24---------------------------- //#35   考虑virtual函数以外的其他选择 { /* 1: ...

  2. Reflux系列01:异步操作经验小结

    写在前面 在实际项目中,应用往往充斥着大量的异步操作,如ajax请求,定时器等.一旦应用涉及异步操作,代码便会变得复杂起来.在flux体系中,让人困惑的往往有几点: 异步操作应该在actions还是s ...

  3. kubernetes部署mysql

    第一章 部署K8S集群 https://www.cnblogs.com/zoulixiang/p/9504324.html 第二章 1.新建mysql-rc.yaml vi mysql-rc.yaml ...

  4. Keyshot+AD渲染PCB效果图

    Keyshot+AD渲染PCB效果图 1.前言 前些天,公司同事找到我说,公司的展会宣传册要更新的了,有几款新的产品需要更新添加上去,大部分的新产品都有实物demo,可以拍照修一下图弄上去.但不巧,其 ...

  5. 如何自出版一本书:定制 bookdown

    目录 如何自出版一本书:定制 bookdown bookdown 的第一步 亚马逊 Kindle 格式 创建书籍 _bookdown.yml 注意行宽 写在每个 .Rmd 文件的开头 index.Rm ...

  6. 用C语言编程自动生成四则运算

    #include<stdio.h>#include<stdlib.h>#include <time.h>#define N 30main(){ int a,b,k, ...

  7. 新安装的Ubuntu设置root密码

    一.问题描述 新安装的Ubuntu切换到root用户时如果没有设置root用户密码会操作失败.此时需要先设置root用户密码. 二.解决办法 打开终端执行 sudo passwd 命令. 输入设置的密 ...

  8. 【菜鸟】RESTful 架构详解

    RESTful 架构详解 分类 编程技术 1. 什么是REST REST全称是Representational State Transfer,中文意思是表述(编者注:通常译为表征)性状态转移. 它首次 ...

  9. OneZero第一次站立会议&Sprint Planning Meeting(2016.3.21)

    会议时间:2016年3月21日 11:40~12:00 会议成员:冉华,张敏,王巍,夏一鸣. 会议目的:列举第一周工作内容,确定第一周工作分配.即挑选出一个Story作为本次迭代完成的目标. 会议内容 ...

  10. Fitts’ Law / 菲茨定律(费茨法则)

     Fitts’ Law / 菲茨定律(费茨法则) 补充一张雅虎ued绘制的关于Fitts’ Law的Q版小漫画,先初步了解下:   Fitts’ Law / 菲茨定律(费茨法则) Fitts’ Law ...