zoukankan      html  css  js  c++  java
  • 【luogu P1471】方差

    https://www.luogu.org/problem/show?pid=1471

    一眼就能看出是线段树/树状数组题目了。

    求平均不用说,线段树/树状数组维护区间和即可。

    方差怎么求?先变换下方差公式:

    可以看到区间的方差可以由区间内每个数的和与每个数的平方的和得来,用一棵线段树维护这两个东西就好了,好像写不了标记永久化。

    当然写两棵普通的线段树/树状数组分别维护这两个东西或者分块暴力也可以不过我写挂了

    区间加的时候如何维护平方的和:

    注意这里的是指没有加之前的和。

    #include <algorithm>
    #include <iostream>
    #define maxn 100005
    using namespace std;
    namespace seg
    {
        struct node
        {
            int ln, rn, mn;
            long double sum[2], mark;
        } seg[maxn * 4];
        void push_down(int p)
        {
            if (seg[p].mark && seg[p].ln != seg[p].rn)
            {
                seg[p * 2].mark += seg[p].mark;
                seg[p * 2].sum[1] += 2 * seg[p].mark * seg[p * 2].sum[0] + (seg[p * 2].rn - seg[p * 2].ln + 1) * seg[p].mark * seg[p].mark;
                seg[p * 2].sum[0] += (seg[p * 2].rn - seg[p * 2].ln + 1) * seg[p].mark;
    
                seg[p * 2 + 1].mark += seg[p].mark;
                seg[p * 2 + 1].sum[1] += 2 * seg[p].mark * seg[p * 2 + 1].sum[0] + (seg[p * 2 + 1].rn - seg[p * 2 + 1].ln + 1) * seg[p].mark * seg[p].mark;
                seg[p * 2 + 1].sum[0] += (seg[p * 2 + 1].rn - seg[p * 2 + 1].ln + 1) * seg[p].mark;
    
                seg[p].mark = 0;
            }
        }
        void init(int l, int r, int p)
        {
            seg[p].ln = l;
            seg[p].rn = r;
            seg[p].mn = (l + r) / 2;
            seg[p].mark = 0;
            if (l != r)
            {
                init(l, seg[p].mn, p * 2);
                init(seg[p].mn + 1, r, p * 2 + 1);
                seg[p].sum[0] = seg[p * 2].sum[0] + seg[p * 2 + 1].sum[0];
                seg[p].sum[1] = seg[p * 2].sum[1] + seg[p * 2 + 1].sum[1];
            }
            else
            {
                cin >> seg[p].sum[0];
                seg[p].sum[1] = seg[p].sum[0] * seg[p].sum[0];
            }
        }
        void increase(int l, int r, long double val, int p)
        {
            if (seg[p].ln == l && seg[p].rn == r)
            {
                seg[p].mark += val;
                seg[p].sum[1] += 2 * val * seg[p].sum[0] + (r - l + 1) * val * val;
                seg[p].sum[0] += (r - l + 1) * val;
            }
            else
            {
                push_down(p);
                if (l <= seg[p].mn)
                    increase(l, min(r, seg[p].mn), val, p * 2);
                if (seg[p].mn + 1 <= r)
                    increase(max(l, seg[p].mn + 1), r, val, p * 2 + 1);
                seg[p].sum[0] = seg[p * 2].sum[0] + seg[p * 2 + 1].sum[0];
                seg[p].sum[1] = seg[p * 2].sum[1] + seg[p * 2 + 1].sum[1];
            }
        }
        long double sum(int l, int r, int p, int emm)
        {
            if (seg[p].ln == l && seg[p].rn == r)
            {
                return seg[p].sum[emm];
            }
            else
            {
                push_down(p);
                long double ans = 0;
                if (l <= seg[p].mn)
                    ans += sum(l, min(r, seg[p].mn), p * 2, emm);
                if (seg[p].mn + 1 <= r)
                    ans += sum(max(l, seg[p].mn + 1), r, p * 2 + 1, emm);
                return ans;
            }
        }
    }
    int n, m;
    int main()
    {
        ios::sync_with_stdio(false);
        cout.precision(4); // 控制精度用
        cin >> n >> m;
        seg::init(1, n, 1);
    
        int opt, l, r;
        long double k;
        long double aver, aver2;
        long double sum, sum2;
        while (m--)
        {
            cin >> opt >> l >> r;
    
            if (opt == 1)
            {
                cin >> k;
                seg::increase(l, r, k, 1);
            }
            else
            {
                sum = seg::sum(l, r, 1, 0);
                sum2 = seg::sum(l, r, 1, 1);
                aver = sum / (long double)(r - l + 1);
                aver2 = sum2 / (long double)(r - l + 1);
                
                if (opt == 2)
                    cout << fixed << aver << endl;
                else
                    cout << fixed << aver2 - aver * aver << endl;
            }
        }
        return 0;
    }
  • 相关阅读:
    vnode之update 还是没太懂
    vnodec创建之标签
    1054 求平均值
    1053 住房空置率
    1052 卖个萌
    1051 复数乘法
    1050 螺旋矩阵
    1049 数列的片段和
    1048 数字加密
    1047 编程团体赛
  • 原文地址:https://www.cnblogs.com/ssttkkl/p/7531894.html
Copyright © 2011-2022 走看看