bzoj 2244 [SDOI2011]拦截导弹(dp+CDQ+树状数组)
题解
看了半天完全没发现这东西和CDQ有什么关系……
先把原序列翻转,求起来方便
然后把每一个位置表示成$(a,b,c)$其中$a$表示位置,$b$表示高度,$c$表示速度,求有多少个位置$a,b,c$都小于它,这就是一个三维偏序问题,直接CDQ就可以解决了……
然后考虑如何求第二问,就是一个导弹所在的LIS数/总的LIS数,因为一个导弹的LIS必须包含自己,以$g[i]$表示以$i$结尾的LIS总数,不难发现有如下转移式
$$g[i]=\sum g[j] \{ (i<j,h[i]<h[j],v[i]<v[j],f[j]+1=f[i]\}$$
这个也可以用CDQ来解决,只要统计出之前最大的LIS再和$f$比较就好了(完全没想出来怎么会这么神仙……)
然后再求出以$i$为开头的LIS长度和方案数,两个相乘就是他自己的方案数了
代码好长……调了好久……
//minamoto
#include<bits/stdc++.h>
using namespace std;
#define getc() (p1==p2&&(p2=(p1=buf)+fread(buf,1,1<<21,stdin),p1==p2)?EOF:*p1++)
char buf[<<],*p1=buf,*p2=buf;
template<class T>inline bool cmax(T&a,const T&b){return a<b?a=b,:;}
inline int read(){
#define num ch-'0'
char ch;bool flag=;int res;
while(!isdigit(ch=getc()))
(ch=='-')&&(flag=true);
for(res=num;isdigit(ch=getc());res=res*+num);
(flag)&&(res=-res);
#undef num
return res;
}
const int N=;
struct BIT{
int f;double w;
BIT(){f=,w=;}
BIT(int f,double w):f(f),w(w){}
}c[N];
int n,st[N],top=;
inline void add(int x,int f,double w){
for(int i=x;i<=n;i+=i&-i){
if(c[i].f<f){
if(c[i].f==) st[++top]=i;
c[i]=(BIT){f,w};
}
else if(c[i].f==f) c[i].w+=w;
}
}
inline BIT query(int x){
BIT res;
for(int i=x;i;i-=i&-i){
if(c[i].f>res.f) res=c[i];
else if(c[i].f==res.f) res.w+=c[i].w;
}
return res;
}
struct node{
int h,v,id,t;
int f[];double g[];
}a[N],q[N];
int wh[N],wv[N],id[N],rk[N];
inline bool cmp(int i,int j)
{return a[i].h<a[j].h||(a[i].h==a[j].h&&a[i].id<a[j].id);}
inline bool cmpid(const node &a,const node &b)
{return a.id<b.id;}
void solve(int l,int r,int t){
if(l==r){
if(a[l].f[t]<) a[l].f[t]=a[l].g[t]=;
return;
}
int mid=(l+r)>>;
memcpy(q+l,a+l,sizeof(node)*(r-l+));
int q1=l,q2=mid+;
for(int i=l;i<=r;++i){
q[i].t<=mid?a[q1++]=q[i]:a[q2++]=q[i];
}
solve(l,mid,t);
q1=l;
for(int i=mid+;i<=r;++i){
while(q1<=mid&&a[q1].id<a[i].id) add(a[q1].v,a[q1].f[t],a[q1].g[t]),++q1;
BIT res=query(a[i].v);
if(res.f==) continue;
if(res.f+>a[i].f[t]){
a[i].f[t]=res.f+,a[i].g[t]=res.w;
}
else if(res.f+==a[i].f[t]) a[i].g[t]+=res.w;
}
while(top){c[st[top--]]=(BIT){,};}
solve(mid+,r,t);
merge(a+l,a+mid+,a+mid+,a+r+,q+l,cmpid);
memcpy(a+l,q+l,sizeof(node)*(r-l+));
}
int th=,tv=;
int main(){
//freopen("testdata.in","r",stdin);
n=read();
for(int i=;i<=n;++i){
wh[i]=a[i].h=read(),wv[i]=a[i].v=read(),a[i].id=i,rk[i]=i;
}
sort(wh+,wh++n),sort(wv+,wv++n);
th=unique(wh+,wh++n)-wh-,tv=unique(wv+,wv++n)-wv-;
for(int i=;i<=n;++i){
a[i].h=th-(lower_bound(wh+,wh+th+,a[i].h)-wh)+;
a[i].v=tv-(lower_bound(wv+,wv+tv+,a[i].v)-wv)+;
}
sort(rk+,rk++n,cmp);
for(int i=;i<=n;++i) a[rk[i]].t=i;
solve(,n,);
for(int i=;i<=n;++i){
a[i].h=th-a[i].h+;
a[i].v=tv-a[i].v+;
a[i].id=n-a[i].id+;
a[i].t=n-a[i].t+;
}
reverse(a+,a++n);
solve(,n,);
reverse(a+,a++n);
double sum=;int ans=;
for(int i=;i<=n;++i){
int len=a[i].f[]+a[i].f[]-;
cmax(ans,len);
}
printf("%d\n",ans);
for(int i=;i<=n;++i){
if(a[i].f[]==ans) sum+=a[i].g[]*a[i].g[];
}
for(int i=;i<=n;++i){
double res=a[i].g[]*a[i].g[];
printf("%.5lf ",(a[i].f[]+a[i].f[]-==ans)?(res/sum):);
}
return ;
}
bzoj 2244 [SDOI2011]拦截导弹(dp+CDQ+树状数组)的更多相关文章
- BZOJ 2244: [SDOI2011]拦截导弹 DP+CDQ分治
2244: [SDOI2011]拦截导弹 Description 某国为了防御敌国的导弹袭击,发展出一种导弹拦截系统.但是这种导弹拦截系统有一个缺陷:虽然它的第一发炮弹能够到达任意的高度.并且能够拦截 ...
- bzoj 2244: [SDOI2011]拦截导弹 cdq分治
2244: [SDOI2011]拦截导弹 Time Limit: 30 Sec Memory Limit: 512 MBSec Special JudgeSubmit: 237 Solved: ...
- BZOJ 2244: [SDOI2011]拦截导弹 (CDQ分治 三维偏序 DP)
题意 略- 分析 就是求最长不上升子序列,坐标取一下反就是求最长不下降子序列,比较大小是二维(h,v)(h,v)(h,v)的比较.我们不看概率,先看第一问怎么求最长不降子序列.设f[i]f[i]f[i ...
- BZOJ 2244: [SDOI2011]拦截导弹 [CDQ分治 树状数组]
传送门 题意:三维最长不上升子序列以及每个元素出现在最长不上升子序列的概率 $1A$了好开心 首先需要从左右各求一遍,长度就是$F[0][i]+F[1][i]-1$,次数就是$G[0][i]*G[1] ...
- BZOJ 1176/2683 Mokia (三维偏序CDQ+树状数组)
题目大意: 洛谷传送门 三维偏序裸题.. 每次操作都看成一个三元组$<x,y,t>$,表示$x,y$坐标和操作时间$t $ 询问操作拆成$4$个容斥 接下来就是$CDQ$了,外层按t排序, ...
- bzoj 2244 [SDOI2011]拦截导弹(DP+CDQ分治+BIT)
[题目链接] http://www.lydsy.com/JudgeOnline/problem.php?id=2244 [题意] 给定n个二元组,求出最长不上升子序列和各颗导弹被拦截的概率. [思路] ...
- BZOJ 2244 [SDOI2011]拦截导弹 (三维偏序CDQ+线段树)
题目大意: 洛谷传送门 不愧为SDOI的duliu题 第一问?二元组的最长不上升子序列长度?裸的三维偏序问题,直接上$CDQ$ 由于是不上升,需要查询某一范围的最大值,并不是前缀最大值,建议用线段树实 ...
- BZOJ 2244 [SDOI2011]拦截导弹 ——CDQ分治
三维偏序,直接CDQ硬上. 正反两次CDQ统计结尾的方案数,最后统计即可. #include <cstdio> #include <cstring> #include < ...
- bzoj 2244: [SDOI2011]拦截导弹
#include<cstdio> #include<iostream> #include<algorithm> #define M 100009 using nam ...
随机推荐
- Java Reference & ReferenceQueue一览
Overview The java.lang.ref package provides more flexible types of references than are otherwise ava ...
- js跳出for循环
1 使用普通的for循环 注意foreach使用return或break都无法跳出循环 2 使用every跳出循环, every 方法会遍历数组中的所有元素来判断是否满足条件,如果有一个元素返回fal ...
- 「这样玩Hexo」修改主题自定义实现界面和功能的自定义
首发于个人博客 想获得更好的阅读体验,烦请移步⬆️ 前言 作为一个颜党,在换了许多Hexo的主题后,选择了现在使用的fexo主题.但是相比于大多数博主使用的NEXT,fexo还是不够powerful, ...
- leetcode 13 Roman to Integer 罗马数组转整型
描述: 将一个字符串表示的罗马数字转为整数,范围0~3999 解决: 如果后一个比前一个大,则表示减,没什么技巧. map<}, {}, {}, {}, {}, {}, {}}; int rom ...
- 滑动窗口的最大值 · sliding-window-maximum
[抄题]: Given an array nums, there is a sliding window of size k which is moving from the very left of ...
- 操作ini配置文件设计一个最基本的可视化数据库系统
对于很多小项目来说,不需要搭建专门的数据库系统(例如用SQLite搭建本地数据库),这时可以用ini配置文件实现一个最基本的数据库,实现数据库最基本的增删改查功能. ini配置文件的用法参考我以前写的 ...
- Vue 路由缓存
问题 在路由切换时不需要每次 点击都刷新子路由 尤其是在form表单的情况下 不能让用户 输入一半之后点击其他页面 再点回来 表单数据不见了 解决方案 vue 2.0 之中 有k ...
- Laravel trait 使用心得
trait 是在PHP5.4中为了方便代码复用的一种实现方式,但目前我在看的的PHP项目中较少看的有程序员去主动使用这个实现方式,在laravel中有很多 trait 的使用,关于trait 在 la ...
- android安装前期遇到的问题
1.安装的eclipse与对应的java版本位数要一致,要么32位,要么64位. 2.关于新版ADT创建项目时出现appcompat_v7的问题 更新ADT至22.6.0版本之后,创建新的安装项目,会 ...
- 3 Django 简介
MVC 与 MTV 模型 MVC Web 服务器开发领域里著名的 MVC 模式,所谓 MVC 就是把 Web 应用分为模型 (M),控制器(C) 和视图 (V) 三层,他们之间以一种插件式的.松耦合的 ...