首页 > 其他 > 详细

codeforce 139E

时间:2018-11-03 20:54:30      阅读:207      评论:0      收藏:0      [点我收藏+]

成段更新+离散化才能过,数据好强。。

单点更新挂在了test27,下次做到成段更新再来做!


/*
期望=存活概率*点权值/100
ans=sum(期望)
离散化树木权值,数轴统计累加可能倒下的树木概率(直接加权值即可)
线段树单点,sum[rt]维护在这个区间中所有蘑菇的存活期望值
*/
#include<iostream>
#include<cstring>
#include<cstdio>
#include<algorithm>
using namespace std;
#define maxn 100005
#define lson l,m,rt<<1
#define rson m+1,r,rt<<1|1
#define ll long long 
using namespace std;
struct Tree{
    int pos,h,pl,pr;//坐标,高度,左概率,右概率
}tree[maxn];
struct Mushroom{
    int pos,w;//蘑菇坐标,蘑菇权值
}mushroom[maxn];
int axis[maxn<<3],cnt,tot;
double poss[maxn<<3];//排序后的坐标点,数轴[1,maxn<<3]上存活概率
double sum[maxn<<2];
void build(int l,int r,int rt){
    sum[rt]=0;
    if(l==r)
        return;
    int m=l+r>>1;
    build(lson);
    build(rson);
}
inline pushup(int rt){
    sum[rt]=sum[rt<<1]+sum[rt<<1|1];
}
void update(int pos,double val,int l,int r,int rt){
    sum[rt]+=val;
    if(l==r) return;
    int m=l+r>>1;
    if(pos<=m) update(pos,val,lson);
    else update(pos,val,rson);
}
int main(){
    int n,m;
    scanf("%d%d",&n,&m);
    cnt=0;
        for(int i=1;i<=n;i++){
            scanf("%d%d%d%d",&tree[i].pos,&tree[i].h,&tree[i].pl,&tree[i].pr);
            axis[cnt++]=tree[i].pos;
            axis[cnt++]=tree[i].pos-tree[i].h;
            axis[cnt++]=tree[i].pos+tree[i].h;
        }
        for(int i=1;i<=m;i++){
            scanf("%d%d",&mushroom[i].pos,&mushroom[i].w);
            axis[cnt++]=mushroom[i].pos;
        }
    
        sort(axis,axis+cnt);
        int tot=unique(axis,axis+cnt)-axis;//离散化
        for(int i=1;i<=tot;i++)//一开始所有的坐标上存活概率都是1
            poss[i]=1;
        //每个点找坐标logn,打标记最多n,复杂度O(n*n)
        for(int i=1;i<=n;i++){
            int pos=lower_bound(axis,axis+tot,tree[i].pos)-axis+1;
            int posl=lower_bound(axis,axis+tot,tree[i].pos-tree[i].h)-axis+1;
            int posr=lower_bound(axis,axis+tot,tree[i].pos+tree[i].h)-axis+1;
            for(int j=posl;j<pos;j++)//倒在左区间的概率
                poss[j]*=1-(double)tree[i].pl/100;
            for(int j=pos+1;j<=posr;j++)//倒在右区间的概率
                poss[j]*=1-(double)tree[i].pr/100;
        }
        //至此统计完概率,开始建立线段树
        build(1,tot,1);
        for(int i=1;i<=m;i++){
            int pos=lower_bound(axis,axis+tot,mushroom[i].pos)-axis+1;
            update(pos,mushroom[i].w*poss[pos],1,tot,1);//把权值*期望更新到线段树中
        }
        printf("%.4lf\n",sum[1]);
    
    return 0;
}

 

 

codeforce 139E

原文:https://www.cnblogs.com/zsben991126/p/9901990.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!