题目链接 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. 使用Myeclipse进行简单webservice开发的示例

    (转) http://blog.csdn.net/changhenshui1990/article/details/70142371 前言:朋友们开始以下教程前,请先看第五大点的注意事项,以避免不必要 ...

  2. Window 10 安装 MySQL

    1.下载 点击官网下载路径下载压缩包 点击第一个按钮后出现如下界面 下载后解压缩,我这里是放在C:\web\目录下 2.创建配置文件 注意:my.ini文件是自己创建的配置文件 文件内容如下: [my ...

  3. 谷歌在线appspot平台教你学Hacker(由浅如深)-XSS篇

    练习链接 http://google-gruyere.appspot.com/ 点开是纯英文的 直接点翻译即可 一 .part1 http://google-gruyere.appspot.com/p ...

  4. 二分查找——Python实现

    一.排序思想 二分(折半)查找思想请参见:https://www.cnblogs.com/luomeng/p/10585291.html 二.python实现 def binarySearchDemo ...

  5. poj 3260 最少硬币(01+多重+完全背包)

    http://www.cnblogs.com/ACMan/archive/2012/08/14/2637437.html #include <iostream> #include < ...

  6. cookie实现记住登录名和密码

    在最近学习的session作用域中,顺便了解了一下cookie, session是存放在服务器中,而cookie是存放在客户端中. 本篇文章主要是使用cookie来实现记住密码的功能. 简单的logi ...

  7. nginx服务器绑定多个域名、支持pathinfo路由、隐藏index.php入口文件

    这篇文章仅仅是操作,解释说明部分待更新. 1. 修改nginx的配置文件(我的配置文件在/etc/nginx/nginx.conf) [root@xxx ~]# find / -name nginx. ...

  8. cf449D. Jzzhu and Numbers(容斥原理 高维前缀和)

    题意 题目链接 给出\(n\)个数,问任意选几个数,它们\(\&\)起来等于\(0\)的方案数 Sol 正解居然是容斥原理Orz,然而本蒟蒻完全想不到.. 考虑每一种方案 答案=任意一种方案 ...

  9. CSS3,3D效果轮播图

    ---恢复内容开始--- 大家还记得我昨天的3D拖拽立方体吗??我昨天还说过css还可以做轮播图,所以咱们今天就写一下,css的轮播图吧! ....这个轮播图主要是用CSS3里的transform的旋 ...

  10. 【Machine Learning】决策树之简介(1)

    Content 1.decision tree representation 2.ID3:a top down learning algorithm 3.expressiveness of data ...