题目链接 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. 跨文件代码跳转插件:Ctags

    1.通过package control搜索Ctags 2.Enter安装,等待其安装完成 3.下载ctags可执行程序,链接:https://pan.baidu.com/s/1jIINAxo 密码:4 ...

  2. RPA流程自动化-UIPath简介

    UiPath简介 转自: http://www.cnblogs.com/mxue/p/UiPath_To147_Road.html 最近RPA比较火,UiPath工具排名前几位并且免费试用,很多朋友们 ...

  3. window.open以post方式提交(转)

    function openWindowWithPost(url,name,keys,values) { var newWindow = window.open(url, name); if (!new ...

  4. golang学习之struct

    结构体定义的一般方式如下: type identifier struct { field1 type1 field2 type2 ... } type T struct {a, b int} 也是合法 ...

  5. 版本控制器之SVN

    1.开发中的实际问题 1.1 小明负责的模块就要完成了,就在即将Release之前的一瞬间,电脑突然蓝屏,硬盘光荣牺牲!几个月来的努力付之东流——需求之一:备份! 1.2 这个项目中需要一个很复杂的功 ...

  6. Mysql根据经纬度筛选数据

    创建位置表,并且插入测试数据 /*     Navicat MySQL Data Transfer     Source Server         : localhost     Source S ...

  7. C# javascript 全选按钮

    function selectAll(checkbox) {                  $('input[type=checkbox]').attr('checked', $(checkbox ...

  8. 什么是APP???APP的开发类型又分哪几种???

    开发者们都知道在高端智能手机系统中有两种应用程序: 一种是基于本地(操作系统)运行的APP —-Native App: 一种是基于高端机的浏览器运行的App —-WebApp因为这些高端智能手机(Ip ...

  9. Express (Routing、Middleware、托管静态文件、view engine 等等)

    1. Express 简介 Express 是基于 Node.js 平台,快速.开放.极简的 web 开发框架,它提供一系列强大的特性,帮助你创建各种 Web 和移动设备应用. Express 不对 ...

  10. 【Web crawler】simulated DFS web crawler

    Finish crawl web learned from udacity 提示:在某些时候,你必须在page上调用get_page.这似乎违反直觉,但是我们用 page 这个词时,指的网页的网址 ( ...