zoukankan      html  css  js  c++  java
  • Luogu P1471 【方差】题解

    我们可以设
    $ sum1[n]=a[1]+a[2]+cdots+a[n] $
    $ sum2[n]=a[1]{2}+a[2]{2}+cdots+a[n]^{2} $

    那么我们就可以表示出方差了

    方差公式展开:

    $ frac {sum_{i=1}^{n} (a[i]-overline{a})^{2} } {n} $

    $ = frac {(a[1]-overline{a})^{2} + (a[2]-overline{a})^{2} + cdots +(a[n]-overline{a})^{2} } {n} $

    $ = frac{a[1]^{2} + a[2]^{2} + cdots +a[n]^{2}- 2 overline{a}(a[1] + a[2] + cdots +a[n] ) +noverline{a}^{2} }{n} $

    $ = frac{a[1]^{2} + a[2]^{2} + cdots +a[n]^{2} }{n} - 2overline{a} frac{a[1]+a[2]+cdots+a[n]}{n}+overline{a}^{2} $

    $ = frac{a[1]^{2} + a[2]^{2} + cdots +a[n]^{2} }{n} - 2overline{a}{2}+overline{a}{2} $

    $ = frac{a[1]^{2} + a[2]^{2} + cdots +a[n]^{2} }{n} - overline{a}^{2} $

    $ = frac{sum2[n]}{n} - overline{a}^{2} $

    至于修改也很简单了

    区间加展开:

    $ sum_{i=1}^{n} (a[i]+x)^{2} $

    $ = (a[1]+x)^{2} + (a[2]+x)^{2} + cdots +(a[n]+x)^{2} $

    $ = a[1]^{2} + a[2]^{2} + cdots +a[n]^{2}+ 2 x(a[1] + a[2] + cdots +a[n] ) +nx^{2} $

    $ = sum2[n]+ 2 x $ $sum1[n] +nx^{2} $

    于是就可以用带lazy的线段树来解决

    *注意:优先级应该是要先处理sum2,再处理sum1

    丑陋的代码如下:

    #include<cstdio>
    
    #define ll long long
    #define lc(x) (x<<1)
    #define rc(x) (x<<1|1)
    
    const ll maxn=100000;
    
    double sum1[(maxn<<2)+5],sum2[(maxn)<<2+5];
    double lazy[(maxn<<2)+5];
    
    inline void pushup(ll v){
        sum1[v]=sum1[lc(v)]+sum1[rc(v)];
        sum2[v]=sum2[lc(v)]+sum2[rc(v)];
    }
    
    inline void pushdown(ll v,ll l,ll r){
        if(lazy[v]){
            ll mid=l+r>>1;
            lazy[lc(v)]+=lazy[v];
            lazy[rc(v)]+=lazy[v];
            sum2[lc(v)]+=2.0*lazy[v]*sum1[lc(v)]+(mid-l+1)*lazy[v]*lazy[v];
            sum2[rc(v)]+=2.0*lazy[v]*sum1[rc(v)]+(r-mid)*lazy[v]*lazy[v];
            sum1[lc(v)]+=lazy[v]*(mid-l+1);
            sum1[rc(v)]+=lazy[v]*(r-mid);
            lazy[v]=0;
        }
    }
    
    void update(ll v,ll l,ll r,ll left,ll right,double add){
        if(left<=l&&r<=right){lazy[v]+=add,sum2[v]+=2.0*sum1[v]*add+(r-l+1)*add*add,sum1[v]+=add*(r-l+1);return;}
        ll mid=l+r>>1;
        pushdown(v,l,r);
        if(left<=mid)update(lc(v),l,mid,left,right,add);
        if(right>mid)update(rc(v),mid+1,r,left,right,add);
        pushup(v);
    }
    
    double query_a(ll v,ll l,ll r,ll left,ll right){
        if(left<=l&&r<=right){return sum1[v];}
        ll mid=l+r>>1;double ans=0;
        pushdown(v,l,r);
        if(left<=mid)ans+=query_a(lc(v),l,mid,left,right);
        if(right>mid)ans+=query_a(rc(v),mid+1,r,left,right);
        return ans;
    }
    
    double query_b(ll v,ll l,ll r,ll left,ll right){
        if(left<=l&&r<=right){return sum2[v];}
        ll mid=l+r>>1;double ans=0;
        pushdown(v,l,r);
        if(left<=mid)ans+=query_b(lc(v),l,mid,left,right);
        if(right>mid)ans+=query_b(rc(v),mid+1,r,left,right);
        return ans;
    }
    
    ll n,m;
    int main(){
        scanf("%lld%lld",&n,&m);
        for(ll i=1;i<=n;i++){
            double p;scanf("%lf",&p);
            update(1,1,n,i,i,p);
        }
        for(ll i=1;i<=m;i++){
            ll op,l,r;scanf("%lld%lld%lld",&op,&l,&r);
            if(op==1){
                double k;scanf("%lf",&k);
                update(1,1,n,l,r,k);
            }
            if(op==2){
                double ans=(double)query_a(1,1,n,l,r)/(r-l+1);
                printf("%.4lf
    ",(double)ans);
            }
            if(op==3){
                double ans=(double)query_a(1,1,n,l,r)/(r-l+1);
                double ans1=(double)query_b(1,1,n,l,r)/(r-l+1);
                double ans2=(double)ans1-(double)ans*ans;
                printf("%.4lf
    ",(double)ans2);
            }
        }
        return 0;
    }
    
  • 相关阅读:
    8-2蒙版初识
    8-1使用自由变换(有些操作和教程不同)
    7-11使用色彩调整图层
    7-10使用历史记录画笔
    7-9将灰度转为彩色
    7-8其他色彩调整
    7-7自动色阶/自动对比度/自动颜色
    7-6替换颜色和色彩范围选取
    7-5匹配颜色
    7-4暗调/高光
  • 原文地址:https://www.cnblogs.com/Railgunforever/p/9965829.html
Copyright © 2011-2022 走看看