题目链接 HDU 1166

大概题意:

第一行一个整数T,表示有T组数据。
每组数据第一行一个正整数N(N<=50000),表示敌人有N个工兵营地,接下来有N个正整数,第i个正整数ai代表第i个工兵营地里开始时有ai个人(1<=ai<=50)。
接下来每行有一条命令,命令有4种形式:
(1) Add i j,i和j为正整数,表示第i个营地增加j个人(j不超过30)
(2)Sub i j ,i和j为正整数,表示第i个营地减少j个人(j不超过30);
(3)Query i j ,i和j为正整数,i<=j,表示询问第i到第j个营地的总人数;
(4)End 表示结束,这条命令在每组数据最后出现;
每组数据最多有40000条命令

思路:类似于区间查询和区间修改等操作,操作数又较多的情况优先想线段树、树状数组等。而线段树和树状数组的相似之处在于二分思想的应用,不同的是前者直接二分,后者要转换为二进制间接对数组以一种特定的组合形式进行二分。

方法一:线段树,因为线段树又是一颗平衡二叉树,所以可以用二叉树的构建方法,在这里用的是结构数组的表示方法。

结点 :T[a, b] (a, b 表示区间 [a, b] , 其中 b-a 为长度 len )

线段树递归定义为:

若 len > 1 , 则 [a, (a+b)/2] 为 T 的左儿子, [(a+b)/2+1, b] 为 T 的右儿子。

若 len == 1, 则 T 为叶子节点。

复杂度:

线段树的深度不超过log2len, 线段树把区间上的任意一条线段都分成不超过 2log2len 条线段。所以线段树能在O(log2len) 时间内完成一条线段的插入, 删除, 和查找等工作。

入门题,AC code:

///HDU 1166 线段树
#include <iostream>
#include <cstdio>
#include <cstring>
using namespace std; struct
{
int a, b, sum; ///左端点, 右端点, 区间和
}t[140000];
int people[50010], SUM; ///每个营地的人数 void make(int x, int y, int num) ///x为左端点,y为右端点,num为数组下标
{
t[num].a = x; ///确定左端点为x
t[num].b = y; ///确定右端点为y if(x == y) ///左端点等于右端点,说明到达叶子结点
t[num].sum = people[y];
else
{
make(x, (x+y)/2, num+num); ///递归构造左子树
make((x+y)/2+1, y, num+num+1); ///递归构造右子树
t[num].sum = t[num+num].sum + t[num+num+1].sum;
///父结点的区间和等于子树的区间和之和,因为区间被分成两半
}
} void add(int i, int j, int num) ///第i个堡垒加j艘船,初始nun为1,即从根结点开始
{
t[num].sum+=j;
if(t[num].a == i && t[num].b == i) ///找到叶子结点,返回
return;
if(i > (t[num].a+t[num].b)/2) ///点i在该区间的右边
add(i, j, num+num+1); ///递归进右结点
else
add(i, j, num+num); ///否则递归进左结点
} void sub(int i, int j, int num) ///第i个堡垒减j艘船
{
t[num].sum-=j;
if(t[num].a == i && t[num].b == i) ///找到叶子结点,返回
return;
if(i > (t[num].a+t[num].b)/2)
sub(i, j, num+num+1);
else
sub(i, j, num+num);
} void query(int i, int j, int num) ///求i到j的总飞船长度, num初始化为1,即从根节点开始
{
if(i <= t[num].a && j >= t[num].b)
SUM+=t[num].sum;
else
{
int mid = (t[num].a + t[num].b)/2;
if(i > mid)
query(i, j, num+num+1);
else if(j <= mid)
query(i, j, num+num);
else
{
query(i, j, num+num);
query(i, j, num+num+1);
}
}
} int main()
{
int N, T;
char command[6];
scanf("%d", &T);
int j = 0;
while(T--)
{
int temp, a, b;
scanf("%d", &N);
for(int i = 1; i <= N; i++)
{
scanf("%d", &people[i]);
}
make(1, N, 1); printf("Case %d:\n", ++j);
while(cin >> command)
{
if(strcmp(command, "End") == 0) break;
else if(strcmp(command, "Query") == 0)
{
cin >> a >> b;
SUM = 0;
query(a, b, 1);
cout << SUM << endl;
}
else if(strcmp(command, "Add") == 0)
{
cin >> a >> b;
add(a, b, 1);
}
else if(strcmp(command, "Sub") == 0)
{
cin >> a >> b;
sub(a, b, 1);
}
}
}
return 0;
}

  

方法二:树状数组

关键在于二进制下的二分思想,理解通过 lowbit (求最低位1)进行数组关系的递推。

Ac code:

///HDU 1166 树状数组
#include <cstdio>
#include <cstring>
#include <iostream>
#include <algorithm>
#define INF 0x3f3f3f3f
using namespace std; const int MAXN = 50005; int N;
int c[MAXN]; ///树状数组 int lowbit(int x) ///位运算,取最低位1,用于后面树状数组下标的二分
{
return x&(-x);
} void add(int i, int value) ///单点加,由上自下更新树状数组
{
while(i <= N)
{
c[i]+=value;
//printf("%d %d\n", i, c[i]);
i+=lowbit(i);
}
} int sum(int i) ///求前缀和
{
int sum = 0;
while(i > 0)
{
sum+=c[i];
i-=lowbit(i);
}
return sum;
} int main()
{
int T;
char command[6];
scanf("%d", &T);
int j = 0;
while(T--)
{
int temp, a, b, d;
scanf("%d", &N);
memset(c, 0, sizeof(c));
for(int i = 1; i <= N; i++)
{
scanf("%d", &d);
add(i, d);
} ///debug
/*
for(int i = 1; i <= N; i++)
printf("%d ", c[i]);
puts("");
*/
printf("Case %d:\n", ++j);
while(cin >> command)
{
if(strcmp(command, "End") == 0) break;
else if(strcmp(command, "Query") == 0)
{
cin >> a >> b;
int SUM = 0;
SUM = sum(b) - sum(a-1);
cout << SUM << endl;
}
else if(strcmp(command, "Add") == 0)
{
cin >> a >> b;
add(a, b);
}
else if(strcmp(command, "Sub") == 0)
{
cin >> a >> b;
add(a, -b);
}
}
}
return 0;
}

  

HDU 1166 【线段树 || 树状数组,单点修改 维护区间和】的更多相关文章

  1. HDU 1166 敌兵布阵 (树状数组 单点修改+区间查询)

    题目链接 Problem Description C国的死对头A国这段时间正在进行军事演习,所以C国间谍头子Derek和他手下Tidy又开始忙乎了.A国在海岸线沿直线布置了N个工兵营地,Derek和T ...

  2. 【算法系列学习】线段树vs树状数组 单点修改,区间查询 [kuangbin带你飞]专题七 线段树 A - 敌兵布阵

    https://vjudge.net/contest/66989#problem/A 单点修改,区间查询 方法一:线段树 http://www.cnblogs.com/kuangbin/archive ...

  3. Libre OJ 130、131、132 (树状数组 单点修改、区间查询 -> 区间修改,单点查询 -> 区间修改,区间查询)

    这三题均可以用树状数组.分块或线段树来做 #130. 树状数组 1 :单点修改,区间查询 题目链接:https://loj.ac/problem/130 题目描述 这是一道模板题. 给定数列 a[1] ...

  4. 牛客小白月赛6 F 发电 树状数组单点更新 求区间乘积 模板

    链接:https://www.nowcoder.com/acm/contest/136/F来源:牛客网  HA实验是一个生产.提炼“神力水晶”的秘密军事基地,神力水晶可以让机器的工作效率成倍提升.   ...

  5. HUST——1106xor的难题之二(异或树状数组单点修改和区间查询)

    1106: xor的难题之二 时间限制: 2 Sec  内存限制: 128 MB 提交: 8  解决: 3 题目描述 上次Alex学长的问题xor难题很简单吧,现在hkhv学长有个问题想问你们. 他现 ...

  6. HDU 1754 I Hate It 【线段树单点修改 维护区间最大值】

    题目传送门:http://acm.hdu.edu.cn/showproblem.php?pid=1754 I Hate It Time Limit: 9000/3000 MS (Java/Others ...

  7. TZOJ 2725 See you~(二维树状数组单点更新区间查询)

    描述 Now I am leaving hust acm. In the past two and half years, I learned so many knowledge about Algo ...

  8. hdu 1166 线段树(sum+单点修改)

    敌兵布阵 Time Limit: 2000/1000 MS (Java/Others)    Memory Limit: 65536/32768 K (Java/Others) Total Submi ...

  9. hdu 2642二维树状数组 单点更新区间查询 模板题

    二维树状数组 单点更新区间查询 模板 从零开始借鉴http://www.2cto.com/kf/201307/227488.html #include<stdio.h> #include& ...

随机推荐

  1. JavaScript中Undefined 和 Null的区别

    Undefined 这个值表示变量不含有值. 可以通过将变量的值设置为 null 来清空变量. 例如: <script> var person; var car="Volvo&q ...

  2. nodejs的jsonrpc调用

    记录下使用nodejs发送jsonrpc请求: var express = require('express'); var router = express.Router(); var request ...

  3. js 表格合并单元格

    5列  根据需要可添加 或 删除 strOneTemp  strTwoTemp  strThreeTemp  strFourTemp  strFiveTemp //合并单元格  this.mergeC ...

  4. 译:Java局部变量类型推断(Var类型)的26条细则

    原文链接:https://dzone.com/articles/var-work-in-progress 作者:Anghel Leonard 译者:沈歌 Java局部变量类型推断(LVTI),简称va ...

  5. git新建分支没有master分支,其他分支也看不到

    git checkout -b dev git新建dev分支,发现切换到了dev分支,但是master分支没有了 git branch和git branch -a 都没有任何反应,看不到其他分支, g ...

  6. python简单的爬虫

    def baidu_tieba(url,begin_page,end_page): for i in range(begin_page, end_page+1): sName = string.zfi ...

  7. Scrapy安装教程 pip 或 conda 两种安装方法.

      cmd: pip -V    查看pip版本 pip install --upgrade pip        升级最高版本 https://sourceforge.net/projects/py ...

  8. ArcGIS DataStore手册——常见问题篇

    第三章:ArcGIS DataStore常见问题处理 1.DataStore使用的数据库是什么? 从安装后的内容和配置完DataStore中Server Manager中的信息来看,DataStore ...

  9. 02_zookeeper配置

    [zoo.cfg] * tickTime:用于计算的时间单元.比如session超时:N*tickTime * initLimit:用于集群,允许从节点连接并且同步到master节点的初始化连接时间, ...

  10. Http重要知识点