[Luogu 3707] SDOI2017 相关分析

<题目链接>


前言

Capella 和 Frank 一样爱好天文学。

她常在冬季的夜晚,若有所思地望着东北方上空的五边形中,最为耀眼的一个顶点。

那一抹金黄曾带给她多少美好的遐想!

直到有一天,她做了这一题,并饱受线段树的折磨。

概述

这是一道区间操作的题目,解法有线段树与分块等。

题主选择线段树进行讲解(因为没用分块写这题)。

准备与计算答案

首先要知道,线段树需要维护的数值有哪些。

先来看题目中所给的公式。

\(\bar x = \frac 1 {r-l+1} \sum_{i=l}^r x_i\)

\(\bar y = \frac 1 {r-l+1} \sum_{i=l}^r y_i\)

\(\hat a = \frac{\sum_{i=l}^r (x_i - \bar x)(y_i - \bar y)}{\sum_{i=l}^r (x_i - \bar x)^2}\)

将线性回归方程的公式展开得(\(\sum_{i=l}^r\) 省略为 \(\sum\)):

\(\hat a = \frac{\sum (x_i y_i - \bar x y_i - \bar y x_i + \bar x \bar y)}
{\sum (x_i^2 - 2\bar x x_i + \bar x^2)}\)

\(= \frac{\sum x_i y_i - \bar x \sum y_i - \bar y \sum x_i + (r-l+1) \bar x \bar y}
{\sum x_i^2 - 2\bar x \sum x_i + (r-l+1) \bar x^2}\)

将 \(\bar x\)、\(\bar y\) 代入上式得:

\(\hat a
= \frac{\sum x_i y_i - \frac{\sum x_i \sum y_i}{r-l+1} - \frac{\sum x_i \sum y_i}{r-l+1} + (r-l+1) \frac{\sum x_i}{r-l+1} \cdot \frac{\sum y_i}{r-l+1}}
{\sum x_i^2 - \frac{2(\sum x_i)^2}{r-l+1} + (r-l+1) (\frac{\sum x_i}{r-l+1})^2}\)

\(= \frac{\sum x_i y_i - \frac{\sum x_i \sum y_i}{r-l+1}}
{\sum x_i^2 - \frac{(\sum x_i)^2}{r-l+1}}\)

由此可见,线段树需要维护的数值有 \(\sum x_i^2\)、\(\sum x_i y_i\)、\(\sum x_i\) 以及 \(\sum y_i\)。

设 \(v_0 = \sum x_i^2, v_1 = \sum x_i y_i, v_2 = \sum x_i, v_3 = \sum y_i\),

则最终计算出的 \(\hat a = \frac{v_1 - \frac{v_2 v_3}{r-l+1}}{v_0 - \frac{v_2^2}{r-l+1}}\)。

区间加操作

进行区间加操作时,设 \(x\) 的变化量为 \(S\),\(y\) 的变化量为 \(T\),则各个标记的下传过程如下:

\(\sum (x_i + S)^2 = \sum (x_i^2 + 2Sx_i + S^2)\)

\(= \sum x_i^2 + 2S \sum x_i + (r-l+1)S^2\)

\(\sum (x_i+S)(y_i+T) = \sum (x_i y_i + Sy_i+Tx_i+ST)\)

\(=\sum x_i y_i + S \sum y_i + T \sum x_i + (r-l+1)ST\)

\(\sum (x_i+S) = \sum x_i + (r-l+1)S\)

\(\sum (y_i+T) = \sum y_i +(r-l+1)T\)

特别注意:由于更新 $\sum x_i^2 $ 与 \(\sum x_i y_i\) 时,需要用到更新前的 \(\sum x_i\) 与 \(\sum y_i\) 的值,所以应注意顺序

区间修改操作

前置技能

\(1^2 + 2^2 + \dots + n^2 = \frac {n(n+1)(2n+1)} 6\)

转化

试图将区间修改操作转化为区间加操作。

对于 \(i \in [l,r]\),\(x_i\) 改为 \(S+i\),\(y_i\) 改为 \(T+i\),

相当于将 \(i \in [l,r]\) 的每个 \(x_i\) 与 \(y_i\) 都改为 \(i\) ,

\(\sum x_i^2 = \sum y_i^2 = \sum i^2 = \frac {r(r+1)(2r+1)} 6 - \frac {l(l-1)(2l-1)} 6\)(即 \(\sum_{i=1}^r i^2 -\sum_{i=1}^{l-1} i^2\))

\(\sum x_i = \sum y_i = \sum i = \frac {(r-l+1)(l+r)} 2\)(\(r-l+1\) 为区间元素个数,\(\frac {l+r} 2\) 为区间平均值)

然后清空Lazy Tag(无论区间曾经加了多少都没有用,将被统一修改)。

维护操作

清空Lazy Tag后,对区间 \([l,r]\) 进行区间加操作,即:每个 \(x_i\) 加上 \(S\),每个 \(y_i\) 加上 \(T\)。

更新过程与区间加操作完全相同。

最后还要打上标记c,表示此区间的子区间需要区间修改操作。

注意事项

  • 一定记得下传Lazy Tagc标记,我就是因为没传才调了一上午+一下午+半个晚上的。

  • 本题下传标记部分较为复杂,请一定注意顺序

结束语

这道题有点像《HAOI2012 高速公路》啊。

线段树这种东西,多推一推也就熟悉啦。

加油,各位队友;加油,自己。

#include <cstdio>
#include <cstring>
const int MAXN=100010;
double x[MAXN],y[MAXN];
int n,m;
class SegmentTree
{
public:
void Build(int i,int l,int r)
{
s[i]=node(l,r);
if(l==r)
{
s[i].v[0]=x[l]*x[l],s[i].v[1]=x[l]*y[r],s[i].v[2]=x[l],s[i].v[3]=y[r];
return;
}
int j=i<<1,mid=l+r>>1;
Build(j,l,mid),Build(j|1,mid+1,r),PushUp(i);
}
void Add(int i,int l,int r,double S,double T)
{
if(l==s[i].l && r==s[i].r)
{
AddModify(i,S,T);
return;
}
if(s[i].l^s[i].r)
PushDown(i);
int j=i<<1,mid=s[i].l+s[i].r>>1;
if(r<=mid)
Add(j,l,r,S,T);
else if(l>mid)
Add(j|1,l,r,S,T);
else
Add(j,l,mid,S,T),Add(j|1,mid+1,r,S,T);
PushUp(i);
}
void Change(int i,int l,int r,double S,double T)
{
if(l==s[i].l && r==s[i].r)
{
ChangeModify(i),AddModify(i,S,T);
return;
}
if(s[i].l^s[i].r)
PushDown(i);
int j=i<<1,mid=s[i].l+s[i].r>>1;
if(r<=mid)
Change(j,l,r,S,T);
else if(l>mid)
Change(j|1,l,r,S,T);
else
Change(j,l,mid,S,T),Change(j|1,mid+1,r,S,T);
PushUp(i);
}
double Ans(double l,double r)
{
double size=r-l+1;
memset(ans,0,sizeof ans);
Sum(1,l,r);
return (ans[1]-ans[2]*ans[3]/size)/(ans[0]-ans[2]*ans[2]/size);
}
private:
double ans[4];
struct node
{
bool c;
double S,T,v[4];
int l,r;
node(int _l=0,int _r=0)
{
l=_l,r=_r,c=S=T=0;
}
}s[MAXN<<2];
double Calc(double i)
{
return i*(i+1)*(2*i+1)/6;
}
void AddModify(int i,double S,double T)
{
double size=double(s[i].r-s[i].l+1);
s[i].v[0]+=S*S*size+2*S*s[i].v[2];
s[i].v[1]+=S*T*size+S*s[i].v[3]+T*s[i].v[2];
s[i].v[2]+=S*size;
s[i].v[3]+=T*size;
s[i].S+=S,s[i].T+=T;
}
void ChangeModify(int i)
{
double l=double(s[i].l),r=double(s[i].r);
s[i].v[0]=s[i].v[1]=Calc(r)-Calc(l-1),s[i].v[2]=s[i].v[3]=(r-l+1)*(l+r)/2,s[i].c=1,s[i].S=s[i].T=0;
}
void PushUp(int i)
{
int l=i<<1,r=l|1;
for(int k=0;k<4;++k)
s[i].v[k]=s[l].v[k]+s[r].v[k];
}
void PushDown(int i)
{
int l=i<<1,r=l|1;
if(s[i].c)
ChangeModify(l),ChangeModify(r);
AddModify(l,s[i].S,s[i].T),AddModify(r,s[i].S,s[i].T);
s[i].c=s[i].S=s[i].T=0;
}
void Sum(int i,int l,int r)
{
if(l==s[i].l && r==s[i].r)
{
for(int k=0;k<4;++k)
ans[k]+=s[i].v[k];
return;
}
if(s[i].l^s[i].r)
PushDown(i);
int j=i<<1,mid=s[i].l+s[i].r>>1;
if(r<=mid)
Sum(j,l,r);
else if(l>mid)
Sum(j|1,l,r);
else
Sum(j,l,mid),Sum(j|1,mid+1,r);
}
}SgT;
int main(int argc,char *argv[])
{
scanf("%d %d",&n,&m);
for(int i=1;i<=n;++i)
scanf("%lf",&x[i]);
for(int i=1;i<=n;++i)
scanf("%lf",&y[i]);
SgT.Build(1,1,n);
for(int i=1,opt,l,r;i<=m;++i)
{
double S,T;
scanf("%d %d %d",&opt,&l,&r);
switch(opt)
{
case 1:
printf("%.10lf\n",SgT.Ans(l,r));
break;
case 2:
scanf("%lf %lf",&S,&T);
SgT.Add(1,l,r,S,T);
break;
case 3:
scanf("%lf %lf",&S,&T);
SgT.Change(1,l,r,S,T);
break;
}
}
return 0;
}

谢谢阅读。

[Luogu 3707] SDOI2017 相关分析的更多相关文章

  1. [题目] Luogu P3707 [SDOI2017]相关分析

    参考资料:[Luogu 3707] SDOI2017 相关分析 P3707 [SDOI2017]相关分析 TFRAC FRAC DFRAC \(\tfrac{\sum}{1}\) \(\frac{\s ...

  2. 洛谷3707 [SDOI2017] 相关分析 【线段树】

    分析: 化简一下就行了,注意一下平方和公式的运用以及精度的误差. 代码: #include<bits/stdc++.h> using namespace std; ; int n,m; i ...

  3. luogu P3707 [SDOI2017]相关分析

    传送门 对于题目要求的东西,考虑拆开懒得拆了 ,可以发现有\(\sum x\sum y\sum x^2\sum xy\)四个变量影响最终结果,考虑维护这些值 下面记\(l,r\)为区间两端点 首先是区 ...

  4. P3707 [SDOI2017]相关分析

    P3707 [SDOI2017]相关分析 线段树裸题?但是真的很麻烦QAQ 题目给的式子是什么不用管,大力拆开,就是\(\frac{\sum x_iy_i-\overline xy_i-\overli ...

  5. BZOJ4817 SDOI2017 相关分析

    4821: [Sdoi2017]相关分析 Time Limit: 10 Sec  Memory Limit: 128 MBSec  Special Judge Description Frank对天文 ...

  6. [Sdoi2017]相关分析 [线段树]

    [Sdoi2017]相关分析 题意:沙茶线段树 md其实我考场上还剩一个多小时写了40分 其实当时写正解也可以吧1h也就写完了不过还要拍一下 正解代码比40分短2333 #include <io ...

  7. 【BZOJ4821】[SDOI2017]相关分析(线段树)

    [BZOJ4821][SDOI2017]相关分析(线段树) 题面 BZOJ 洛谷 题解 看看询问要求的东西是什么.把所有的括号拆开,不难发现要求的就是\(\sum x,\sum y,\sum xy,\ ...

  8. 4821: [Sdoi2017]相关分析

    4821: [Sdoi2017]相关分析 链接 分析: 大力拆式子,化简,然后线段树.注意精度问题与爆longlong问题. 代码: #include<cstdio> #include&l ...

  9. AC日记——[SDOI2017]相关分析 洛谷 P3707

    [SDOI2017]相关分析 思路: 裸线段树: (玄学ac): 代码: #include <bits/stdc++.h> using namespace std; #define max ...

随机推荐

  1. [javascript] 看知乎学习js闭包

    知乎:到底什么是闭包? 寸志: JavaScript 闭包的本质源自两点,词法作用域和函数当作值传递. 词法作用域,就是,按照代码书写时的样子,内部函数可以访问函数外面的变量.引擎通过数据结构和算法表 ...

  2. 【RabbitMQ】1、RabbitMQ的几种典型使用场景

    RabbitMQ主页:https://www.rabbitmq.com/ AMQP AMQP协议是一个高级抽象层消息通信协议,RabbitMQ是AMQP协议的实现.它主要包括以下组件: 1.Serve ...

  3. HDU6198

    number number number Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Oth ...

  4. HDU3062(2-SAT)

    Party Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K (Java/Others)Total Submi ...

  5. layer弹出框确定前验证:弹出消息框(弹出两个layer)

    作者QQ:1095737364 QQ群:123300273 欢迎加入! layer 弹出框中经常遇到要弹出表单进行修改数据, 因此在弹出框中的表单需要验证数据, 就需要在弹出一个layer, 默认的设 ...

  6. js-ES6学习笔记-编程风格(1)

    1.ES6提出了两个新的声明变量的命令:let和const.其中,let完全可以取代var,因为两者语义相同,而且let没有副作用. 2.var命令存在变量提升效用,let命令没有这个问题.建议不再使 ...

  7. <Android 基础(三十四)> TabLayout 从头到脚

    1. 简介 1.TabLayout给我们提供的是一排横向的标签页 2.#newTab()这个方法来创建新的标签页,然后用过#setText()和#setIcon方法分别修改标签页的文本和图标,创建完成 ...

  8. pgAdmin4 ubuntu python 安装

    ubuntu安装pgAdmin4,通过python的pip 安装 pgAdmin4.(首更时间20161205) 新版本的pgAdmin4目前支持mac/window/linux/python,可是l ...

  9. FI配置步骤清单

    1.定义公司代码   配置路径: R/3定制IMG的实施指南>企业结构>定义>财务会计>定义, 复制, 删除, 检查公司代码 事务代码 EC01 2.编辑科目表清单   配置路 ...

  10. KCF跟踪算法 入门详解

    一.算法介绍 KCF全称为Kernel Correlation Filter 核相关滤波算法.是在2014年由Joao F. Henriques, Rui Caseiro, Pedro Martins ...