牛客练习赛28-B(线段树,区间更新)
牛客练习赛28 - B
题目
qn姐姐最好了~
qn姐姐给你了一个长度为n的序列还有m次操作让你玩,
1 l r 询问区间[l,r]内的元素和
2 l r 询问区间[l,r]内的
元素的平方
和
3 l r x 将区间[l,r]内的每一个元素都乘上x
4 l r x 将区间[l,r]内的每一个元素都加上x
输入描述:
第一行两个数n,m
接下来一行n个数表示初始序列
就下来m行每行第一个数为操作方法opt,
若opt=1或者opt=2,则之后跟着两个数为l,r
若opt=3或者opt=4,则之后跟着三个数为l,r,x
操作意思为题目描述里说的
输出描述:
对于每一个操作1,2,输出一行表示答案
输入
复制
5 6
1 2 3 4 5
1 1 5
2 1 5
3 1 2 1
4 1 3 2
1 1 4
2 2 3
输出
复制
15
55
16
41
备注:
对于100%的数据 n=10000,m=200000 (注意是等于号)
保证所有询问的答案在long long 范围内
虽然这道题暴力能卡过去,真的是卡过去的,什么乱七八糟的宏定义、全局变量都要删了。。。
线段树做法:主要是处理x[i] * x[i]的前 n 项和,这里sum[][0]表示前区间和,sum[][1]表示区间内的元素平方和。
这里x = mul * x + add;下面是做各种操作后sum及懒惰标记的变化
乘法 * num:
mul[rt] = mul[rt] * num
add[rt] = add[rt] * num (如果add = 0,那么add * num = 0,如果add = a, 那么 x = (mul * x + a) * num,即 x = mul * num * x + a * num,按照x = mul * x + add的格式)
sum[][0] = sum[][0] * num
sum[][1] = sum[][1] * num * num
加法 + num:
add[rt] += num
sum[][1] += 2 * sum[][0] * num + (r - l + 1) * num * num
这里的sum[][0]是还没改变的,即这里先处理sum[][1],下面再处理sum[][0]。先来看单个元素的变化,(x + num) * (x + num) = x * x + 2 * num * x + num * num, 然后多个这样的式子相加,即可得上式,也可以解释为什么先处理sum[][1]
sum[][0] += (r - l + 1) * num
最后要记得long long,查询和更新都要下推标记,mul[]要初始化
#include <iostream>
#include <cstdio>
#include <string>
#include <cstring>
#include <cmath>
#include <queue>
#include <iomanip>
#include <stack>
#include <algorithm>
#include <vector>
#include <functional>
using namespace std;
typedef long long LL;
#define Fil(Array, len, num) fill(Array, Array + len, num)
#define lson l, m, rt << 1
#define rson m + 1, r, rt << 1 | 1
const double PI = 3.1415926;
const double E = 2.1718281828;
const int MAXN = 10005;
const int INF = 0x3f3f3f3f;
const int MOD = 1e9 + 7;
LL sum[MAXN << 2][3], add[MAXN << 2], mul[MAXN << 2];
void PushUp(LL rt)
{
sum[rt][0] = sum[rt << 1][0] + sum[rt << 1 | 1][0];
sum[rt][1] = sum[rt << 1][1] + sum[rt << 1 | 1][1];
}
void PushDown(LL ln, LL rn, LL rt)
{
if(mul[rt] != 1)
{
mul[rt << 1] *= mul[rt], mul[rt << 1 | 1] *= mul[rt];
sum[rt << 1][1] *= mul[rt] * mul[rt], sum[rt << 1 | 1][1] *= mul[rt] * mul[rt];
sum[rt][0] *= mul[rt], sum[rt << 1 | 1][0] *= mul[rt];
mul[rt] = 1;
}
if(add[rt])
{
add[rt << 1] += add[rt], add[rt << 1 | 1] += add[rt];
sum[rt << 1][1] += 2 * sum[rt << 1][0] * add[rt] + ln * add[rt] * add[rt];
sum[rt << 1 | 1][1] += 2 * sum[rt << 1 | 1][0] * add[rt] + rn * add[rt] * add[rt];
sum[rt << 1][0] += ln * add[rt], sum[rt << 1 | 1][0] += rn * add[rt];
add[rt] = 0;
}
}
void Build(LL l, LL r, LL rt)
{
add[rt] = 0, mul[rt] = 1;
if(l == r)
{
scanf("%lld", &sum[rt][0]);
sum[rt][1] = sum[rt][0] * sum[rt][0];
return ;
}
LL m = (l + r) >> 1;
Build(lson);
Build(rson);
PushUp(rt);
}
void Updata_Add(LL L, LL R, LL num, LL l, LL r, LL rt)
{
if(L <= l && r <= R)
{
sum[rt][1] += 2 * sum[rt][0] * num + (r - l + 1) * num * num;
sum[rt][0] += (r - l + 1) * num;
add[rt] += num;
return ;
}
LL m = (l + r) >> 1;
PushDown(m - l + 1, r - m, rt);
if(L <= m)
Updata_Add(L, R, num, lson);
if(m < R)
Updata_Add(L, R, num, rson);
PushUp(rt);
}
void Updata_mul(LL L, LL R, LL num, LL l, LL r, LL rt)
{
if(L <= l && r <= R)
{
sum[rt][0] *= num;
sum[rt][1] *= num * num;
mul[rt] *= num;
add[rt] *= num;
return ;
}
LL m = (l + r) >> 1;
PushDown(m - l + 1, r - m, rt);
if(L <= m)
Updata_mul(L, R, num, lson);
if(m < R)
Updata_mul(L, R, num, rson);
PushUp(rt);
}
LL Query(LL L, LL R, LL l, LL r, LL rt, LL opt)
{
if(L <= l && r <= R)
{
if(opt == 1)
return sum[rt][0];
if(opt == 2)
return sum[rt][1];
}
LL ans = 0;
LL m = (l + r) >> 1;
PushDown(m - l + 1, r - m, rt);
if(L <= m)
ans += Query(L, R, lson, opt);
if(m < R)
ans += Query(L, R, rson, opt);
return ans;
}
int main()
{
Fil(mul, (MAXN << 2), 1);
LL n, m;
scanf("%d%d", &n, &m);
Build(1, n, 1);
while(m--)
{
LL opt, l, r, x;
scanf("%d%d%d", &opt, &l, &r);
if(opt == 1 || opt == 2)
printf("%lld\n", Query(l, r, 1, n, 1, opt));
else if(opt == 3)
{
scanf("%lld", &x);
Updata_mul(l, r, x, 1, n, 1);
}
else
{
scanf("%lld", &x);
Updata_Add(l, r, x, 1, n, 1);
}
}
return 0;
}
牛客练习赛28-B(线段树,区间更新)的更多相关文章
- hdu 3966(树链剖分+线段树区间更新)
传送门:Problem 3966 https://www.cnblogs.com/violet-acmer/p/9711441.html 学习资料: [1]线段树区间更新:https://blog.c ...
- hdu 1698 线段树 区间更新 区间求和
Just a Hook Time Limit: 4000/2000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others)Total ...
- HDU3577 线段树(区间更新)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=3577 ,普通的线段树区间更新题目,较简单. 相当于一个区间覆盖问题,有一点要注意的就是叶子节点是一个长 ...
- HDU 1556 Color the ball(线段树区间更新)
Color the ball 我真的该认真的复习一下以前没懂的知识了,今天看了一下线段树,以前只会用模板,现在看懂了之后,发现还有这么多巧妙的地方,好厉害啊 所以就应该尽量搞懂 弄明白每个知识点 [题 ...
- hihoCoder 1080 : 更为复杂的买卖房屋姿势 线段树区间更新
#1080 : 更为复杂的买卖房屋姿势 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 小Hi和小Ho都是游戏迷,“模拟都市”是他们非常喜欢的一个游戏,在这个游戏里面他们 ...
- HDU 5023 A Corrupt Mayor's Performance Art(线段树区间更新)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=5023 解题报告:一面墙长度为n,有N个单元,每个单元编号从1到n,墙的初始的颜色是2,一共有30种颜色 ...
- HDU 4902 Nice boat 2014杭电多校训练赛第四场F题(线段树区间更新)
题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4902 解题报告:输入一个序列,然后有q次操作,操作有两种,第一种是把区间 (l,r) 变成x,第二种是 ...
- HDU 1698 线段树 区间更新求和
一开始这条链子全都是1 #include<stdio.h> #include<string.h> #include<algorithm> #include<m ...
- POJ-2528 Mayor's posters (线段树区间更新+离散化)
题目分析:线段树区间更新+离散化 代码如下: # include<iostream> # include<cstdio> # include<queue> # in ...
随机推荐
- ubuntu 12.04安装vmtools 问题解决
vmware安装ubuntu12.04版本的vm tools时 遇到: Searching for a valid kernel header path... The path "& ...
- Grunt 与WebStrom 集成
为了不想使用命令行的方式开着grunt,打算将Grunt命令集成WebStrom 中 . 1.将配置好的Gruntfile文件放到项目的根目录下.. 2.File-setting-Extental T ...
- 关于super关键字与继承
super它只是一个限定词,当用super引用时,它也是引用当前对象本身,只是super只是限定了访问当前对象从父类那里继承得到成员变量或方法. import java.util.Date; publ ...
- sqlite3使用详解(Qt版本)
初始化sqlite3 (创建表) QString url = QDir::currentPath() + QString::fromLocal8Bit("/Msg.db"); bo ...
- Java网络编程のOSI
我们可以把客户机和远程服务器理解为主机A和主机B,用户和主机A可以通过主机A中的应用程序进行交互,主机A与主机B之间交互则是通过计算机网络通信进行的. 网络中每台机器称为节点.大多数节点是计算机,此外 ...
- Fiddler手机https抓包
Fiddler手机抓包:https://blog.csdn.net/wangjun5159/article/details/52202059 fiddler 使用说明:https://www.cnbl ...
- Java50道经典习题-程序4 分解质因数
题目:将一个正整数分解质因数.例如:输入90,打印出90=2*3*3*5.分析:对n进行分解质因数,应先找到一个最小的质数k,然后按下述步骤完成:(1)如果这个质数恰等于n,则说明分解质因数的过程已经 ...
- uwsgi怎么启动停止
## 二.启动停止重启 uWSGI 通过 xxx.ini 启动后会在相同目录下生成一个 xxx.pid 的文件,里面只有一行内容是 uWSGI 的主进程的进程号. uWSGI 启动:uwsgi --i ...
- 提示缺少python.h解决办法
在安装uwsgi时,提示缺少python.h In file included :: plugins/python/uwsgi_python.h::: fatal error: Python.h: N ...
- base64编码问题
最近遇到一个很奇怪的问题:post方式上传文件,因为文件不大,所以直接base64后作为参数扔给服务器.一开始好用,后来出问题了,上传的压缩包再下载后,能双击打开看到压缩包里面的文件,但是解压就报错, ...