题目

给定一组数,要求进行若干次操作,这些操作可以分为两种类型: 
(1) CMD 1 beg end value 将数组中下标在[beg, end] 区间内数字都变为value 
(2) CMD 2 beg end 求出数组中下标在[beg ,end]区间中的所有数字的和

分析

树状数组在区间查询和单点修改情况下效率较线段树高一些,而无法像线段树一样在O(logN)的时间内完成区间修改。因此使用线段树解决。使用线段树主要的是定义好线段树节点的状态。(如代码中注释所说,状态一定要明确,且容易计算!)

实现

#include<iostream>
#include<string.h>
#include<iostream>
#include<queue>
#include<cmath>
#include<unordered_map>
#include<unordered_set>
#include<string>
#include<vector>
using namespace std;
const int inf = 1 << 29;
const int kMax = 100005;
struct Node{
int beg;
int end;
int val;
//若val为非零值,表示当前时刻,节点所代表的区间内所有的值同时被修改为val;
//如果为0,一种情况是该节点代表区间内的值同时被修改为val,
//一种情况是:该区间内的值没有被同时修改为val(可能从没被修改过,或者之前被同时修改过,但是后来又被修改了其中一部分) int sum; //当前时刻,区间内所有数字的和。这个值就是当前时刻的值,不需要参考value
Node(){
val = sum = 0;
}
};
Node gNodes[4 * kMax];
int weight[kMax];
void BuildTree(int node, int beg, int end){
gNodes[node].beg = beg;
gNodes[node].end = end;
if (beg == end){
gNodes[node].val = gNodes[node].sum = weight[beg]; //初始化
return;
}
int left = 2 * node + 1, right = 2 * node + 2;
int mid = (beg + end) / 2;
BuildTree(left, beg, mid);
BuildTree(right, mid + 1, end);
gNodes[node].sum = gNodes[left].sum + gNodes[right].sum; }
//从上向下更新
void PushDown(int node){
if (gNodes[node].beg == gNodes[node].end){
//叶子节点处的更新,注意,由于 线段树的节点中的 sum被定义为当前时刻的真实的和。那么,
//当叶子节点被修改为了value时,同时将sum给计算出来
gNodes[node].sum = gNodes[node].val;
return;
} int left = 2 * node + 1, right = 2 * node + 2;
if (gNodes[node].val){
//注意,由于 线段树的节点中的 sum被定义为当前时刻的真实的和。那么,每当pushdown,子节点的value被修改时,
//也需要同时将 sum给计算出来!
int value = gNodes[node].val;
gNodes[left].val = gNodes[right].val = value;
gNodes[left].sum = (gNodes[left].end - gNodes[left].beg + 1)*value;
gNodes[right].sum = (gNodes[right].end - gNodes[right].beg + 1)*value;
}
gNodes[node].val = 0;
} //从下向上更新
void PushUp(int node){
if (gNodes[node].beg == gNodes[node].end){
gNodes[node].sum = gNodes[node].val;
return;
} int left = 2 * node + 1, right = 2 * node + 2;
gNodes[node].sum = gNodes[left].sum + gNodes[right].sum;
} void Update(int node, int beg, int end, int value){
if (beg == gNodes[node].beg && end == gNodes[node].end){
//对区间进行更新,节点的val 更新为value不用说了。注意由于我们定义的 节点中的sum为当前时刻的真实的和,因此
//需要实时的计算出来
gNodes[node].val = value;
gNodes[node].sum = gNodes[node].val*(gNodes[node].end - gNodes[node].beg + 1);
return;
}
if (beg > end)
return;
//查询区间和线段树节点代表的区间不同,则进行区间分解。 需要先将父节点的信息传递给子节点
PushDown(node);
int left = 2 * node + 1, right = 2 * node + 2;
int mid = (gNodes[node].beg + gNodes[node].end) / 2;
if (mid >= end){
Update(left, beg, end, value);
}
else if(mid < beg){
Update(right, beg, end, value);
}
else{
Update(left, beg, mid, value);
Update(right, mid + 1, end, value);
}
//线段树子节点更新完之后,需要更新父节点的 sum 信息
PushUp(node);
} int Query(int node, int beg, int end){
if (gNodes[node].beg == beg && gNodes[node].end == end){
return gNodes[node].sum;
}
if (beg > end)
return 0;
PushDown(node);
int left = 2 * node + 1, right = 2 * node + 2;
int mid = (gNodes[node].beg + gNodes[node].end) / 2;
if (mid >= end){
return Query(left, beg, end);
}
else if (mid < beg){
return Query(right, beg, end);
}
else{
int left_sum = Query(left, beg, mid);
int right_sum = Query(right, mid + 1, end);
return left_sum + right_sum;
}
}
int main(){
int n;
scanf("%d", &n);
for (int i = 0; i < n; i++){
scanf("%d", &weight[i]);
}
BuildTree(0, 0, n - 1);
scanf("%d", &n);
int cmd, beg, end, value;
for (int i = 0; i < n; i++){
scanf("%d", &cmd);
if (cmd == 0){
scanf("%d %d", &beg, &end);
int result = Query(0, beg - 1, end - 1);
printf("%d\n", result);
}
else{
scanf("%d %d %d", &beg, &end, &value);
Update(0, beg - 1, end-1, value);
}
}
return 0;
}

hiho_1078_线段树区间修改的更多相关文章

  1. Codeforces Round #442 (Div. 2) E Danil and a Part-time Job (dfs序加上一个线段树区间修改查询)

    题意: 给出一个具有N个点的树,现在给出两种操作: 1.get x,表示询问以x作为根的子树中,1的个数. 2.pow x,表示将以x作为根的子树全部翻转(0变1,1变0). 思路:dfs序加上一个线 ...

  2. 题解报告:hdu 1698 Just a Hook(线段树区间修改+lazy懒标记的运用)

    Problem Description In the game of DotA, Pudge’s meat hook is actually the most horrible thing for m ...

  3. poj 2528 线段树区间修改+离散化

    Mayor's posters POJ 2528 传送门 线段树区间修改加离散化 #include <cstdio> #include <iostream> #include ...

  4. E - Just a Hook HDU - 1698 线段树区间修改区间和模版题

    题意  给出一段初始化全为1的区间  后面可以一段一段更改成 1 或 2 或3 问最后整段区间的和是多少 思路:标准线段树区间和模版题 #include<cstdio> #include& ...

  5. HDU 4027 Can you answer these queries? (线段树区间修改查询)

    描述 A lot of battleships of evil are arranged in a line before the battle. Our commander decides to u ...

  6. poj2528 Mayor's posters(线段树区间修改+特殊离散化)

    Description The citizens of Bytetown, AB, could not stand that the candidates in the mayoral electio ...

  7. HDU - 1698 线段树区间修改,区间查询

    这就是很简单的基本的线段树的基本操作,区间修改,区间查询,对区间内部信息打上laze标记,然后维护即可. 我自己做的时候太傻逼了...把区间修改写错了,对给定区间进行修改的时候,mid取的是节点的左右 ...

  8. HDU - 3974 Assign the task (线段树区间修改+构建模型)

    https://cn.vjudge.net/problem/HDU-3974 题意 有一棵树,给一个结点分配任务时,其子树的所有结点都能接受到此任务.有两个操作,C x表示查询x结点此时任务编号,T ...

  9. SPOJ GSS2 - Can you answer these queries II(线段树 区间修改+区间查询)(后缀和)

    GSS2 - Can you answer these queries II #tree Being a completist and a simplist, kid Yang Zhe cannot ...

随机推荐

  1. Unity-Animator深入系列---StateMachineBehaviour初始化时间测试

    回到 Animator深入系列总目录 结果和想的有点出入 测试结果: 1.SMB初始化会被调用多次,次数不可控,当Animator组件重复开关则重复初始化. 2.SMB支持构造函数 MyClass p ...

  2. HDU(2089),数位DP

    题目链接:http://acm.split.hdu.edu.cn/showproblem.php?pid=2089 不要62 Time Limit: 1000/1000 MS (Java/Others ...

  3. php cli模式学习(PHP命令行模式)

    http://www.jb51.net/article/37796.htm php_cli模式简介  php-cli是php Command Line Interface的简称,如同它名字的意思,就是 ...

  4. Java多线程的三种实现方式

    java多线程的三种实现方式 一.继承Thread类 二.实现Runnable接口 三.使用ExecutorService, Callable, Future 无论是通过继承Thread类还是实现Ru ...

  5. 让VS自动生成我们自己的注释

    1. 找到你VS的安装目录:C:\Program Files (x86)\Microsoft Visual Studio 11.0 2. 在VS安装路径下依次找到这些文件夹:\Common7\IDE\ ...

  6. iOS中3种正则表达式的使用与比较

    正则表达式在用户注册和登录中应用很广,通过正则表达式可以判断用户输入的数据正确与否. 在iOS4.0以前开发者一般是通过谓词(NSPredicate)和加入正则表达式的第三方库(如:RegexKitL ...

  7. JS中string对象的一些方法

    原文地址(包含所有的string对象的方法):  http://www.dreamdu.com/javascript/object_string/ string.slice(startPos,endP ...

  8. vsftpd配置参数详细整理

    vsftpd配置参数详细整理  -|白王斧三又干一 vsftpd配置参数详细整理     -|白王斧三又干一 发表于 2005-10-23 20:30:00   1.vsftpd配置参数详细整理#接受 ...

  9. 获取DIV与浏览器顶部相聚一定位置之后移动DIV

    获取元素(这里定位元素A)距离顶部的高度,接着设定scroll滚动的事件,比如超过那个高度,把A的位置设定为fixed,小于该高度,修改回relative. 方法一: $(function() {  ...

  10. C# 十进制与十六进制互转

    1.从十六进制转换为十进制 /// <summary> /// 十六进制转换到十进制 /// </summary> /// <param name="hex&q ...