Mobile phones
Time Limit: 5000MS   Memory Limit: 65536K
Total Submissions: 14489   Accepted: 6735

Description

Suppose that the fourth generation mobile phone base stations in the Tampere area operate as follows. The area is divided into squares. The squares form an S * S matrix with the rows and columns numbered from 0 to S-1. Each square contains a base station. The number of active mobile phones inside a square can change because a phone is moved from a square to another or a phone is switched on or off. At times, each base station reports the change in the number of active phones to the main base station along with the row and the column of the matrix.

Write a program, which receives these reports and answers queries about the current total number of active mobile phones in any rectangle-shaped area.

Input

The input is read from standard input as integers and the answers to the queries are written to standard output as integers. The input is encoded as follows. Each input comes on a separate line, and consists of one instruction integer and a number of parameter integers according to the following table. 

The values will always be in range, so there is no need to check them. In particular, if A is negative, it can be assumed that it will not reduce the square value below zero. The indexing starts at 0, e.g. for a table of size 4 * 4, we have 0 <= X <= 3 and 0 <= Y <= 3.

Table size: 1 * 1 <= S * S <= 1024 * 1024 
Cell value V at any time: 0 <= V <= 32767 
Update amount: -32768 <= A <= 32767 
No of instructions in input: 3 <= U <= 60002 
Maximum number of phones in the whole table: M= 2^30 

Output

Your program should not answer anything to lines with an instruction other than 2. If the instruction is 2, then your program is expected to answer the query by writing the answer as a single line containing a single integer to standard output.

Sample Input

  1. 0 4
  2. 1 1 2 3
  3. 2 0 0 2 2
  4. 1 1 1 2
  5. 1 1 2 -1
  6. 2 1 1 2 3
  7. 3

Sample Output

  1. 3
  2. 4

Source

 
  二维树状数组,经典题
  这道题用二维线段树做也可以,做法见右方链接poj 1195:Mobile phones(二维线段树,矩阵求和)
  二维树状数组和二维线段树效率对比(上为树状数组,下为线段树):
 
  题意

  一个矩阵,初始化为全0。有以下操作:
  1)将 (x,y) 元素加 a。x为行坐标,y为列坐标。
  2)求矩阵 [X,Y] 所有元素的和。该矩阵左上角为[l,b],右下角为[r,t]。
 
  思路
  同一维的树状数组求法差不多。
  分别说明二维树状数组加数和求和的做法:
  1)加数:在矩阵的某个坐标处加上某个数,则二维树状数组中的值也要改变。那么改变哪些位置的值呢?我们知道,一维的加数是不断改变在这个位置之上的所有位置的值,之上的位置是用lowbit()求出来的。例如要改变的位置是x,你要加的数是a,则需要把x不断加上lowbit(),在新的x的位置加上a,然后再将x加lowbit(),再在x上加a……直到x超过这个数组规定的大小。
  二维的也是类似,只不过在计算每一个x的时候,需要再计算一个y。例如要在(x,y)位置加上一个数a。先保持x不变,y不断加lowbit(),在对应的(x,y)的位置加a,直到y超过限界(如果矩阵是4*4的,则这个限界就是4,y<=4)。一轮过后,将x加lowbit(),y恢复一开始的值,再进行新的一轮,这样一轮一轮的直到x也超过限界(同样如果矩阵大小为4*4,则x<=4)。
  如此这般,假设矩阵大小为4*4,则Add的代码如下:
  1. void Add(int x,int y,int a)
  2. {
  3. int i=x;
  4. while(i<=s){
  5. int j=y;
  6. while(j<=s){
  7. c[i][j]+=a;
  8. j+=lowbit(j);
  9. }
  10. i+=lowbit(i);
  11. }
  12. }

  2)求和:对左上角为(l,r),右下角为(b,t)的矩阵求和,即求该矩阵中所有元素的和。先求l到r的行矩阵的和,在求这个行矩阵和的时候,每一行要计算对应的b到t列的元素和。具体过程类似上述过程,将+lowbit()改为-lowbit()即可。限界为>=1。

  

  代码

  1. #include <iostream>
  2. #include <stdio.h>
  3. #include <string.h>
  4. using namespace std;
  5.  
  6. #define MAXN 1100
  7.  
  8. int c[MAXN][MAXN],s;
  9.  
  10. int lowbit(int x)
  11. {
  12. return x&-x;
  13. }
  14.  
  15. void Add(int x,int y,int a) //加数
  16. {
  17. int i=x;
  18. while(i<=s){ //行
  19. int j=y;
  20. while(j<=s){ //列
  21. c[i][j]+=a;
  22. j+=lowbit(j);
  23. }
  24. i+=lowbit(i);
  25. }
  26. }
  27.  
  28. int Sum(int l,int r,int b,int t) //求和
  29. {
  30. l--,b--;
  31. int suml=,sumr=;
  32. //求行矩阵和,l以上矩阵
  33. while(l>=){
  34. int i=b,j=t;
  35. int sumb=,sumt=;
  36. //求列矩阵和
  37. while(i>=){
  38. sumb+=c[l][i];
  39. i-=lowbit(i);
  40. }
  41. while(j>=){
  42. sumt+=c[l][j];
  43. j-=lowbit(j);
  44. }
  45. suml+=sumt-sumb;
  46. l-=lowbit(l);
  47. }
  48. //求行矩阵和,r以上矩阵
  49. while(r>=){
  50. int i=b,j=t;
  51. int sumb=,sumt=;
  52. //求列矩阵和
  53. while(i>=){
  54. sumb+=c[r][i];
  55. i-=lowbit(i);
  56. }
  57. while(j>=){
  58. sumt+=c[r][j];
  59. j-=lowbit(j);
  60. }
  61. sumr+=sumt-sumb;
  62. r-=lowbit(r);
  63. }
  64. return sumr-suml;
  65. }
  66.  
  67. int main()
  68. {
  69. int cmd,x,y,a,l,r,b,t;
  70.  
  71. while(scanf("%d",&cmd)!=EOF){
  72. switch(cmd){
  73. case : //初始化矩阵
  74. scanf("%d",&s);
  75. memset(c,,sizeof(c));
  76. break;
  77.  
  78. case : //加数
  79. scanf("%d%d%d",&x,&y,&a);
  80. Add(x+,y+,a);
  81. break;
  82.  
  83. case : //求矩阵和
  84. scanf("%d%d%d%d",&l,&b,&r,&t);
  85. printf("%d\n",Sum(l+,r+,b+,t+));
  86. break;
  87.  
  88. case : //退出程序
  89. return ;
  90. default:
  91. break;
  92. }
  93. }
  94. return ;
  95. }

Freecode : www.cnblogs.com/yym2013

poj 1195:Mobile phones(二维树状数组,矩阵求和)的更多相关文章

  1. poj 1195 Mobile phones(二维树状数组)

    树状数组支持两种操作: Add(x, d)操作:   让a[x]增加d. Query(L,R): 计算 a[L]+a[L+1]……a[R]. 当要频繁的对数组元素进行修改,同时又要频繁的查询数组内任一 ...

  2. POJ 1195:Mobile phones 二维树状数组

    Mobile phones Time Limit: 5000MS   Memory Limit: 65536K Total Submissions: 16893   Accepted: 7789 De ...

  3. 【poj1195】Mobile phones(二维树状数组)

    题目链接:http://poj.org/problem?id=1195 [题意] 给出一个全0的矩阵,然后一些操作 0 S:初始化矩阵,维数是S*S,值全为0,这个操作只有最开始出现一次 1 X Y ...

  4. POJ 2155 Matrix【二维树状数组+YY(区间计数)】

    题目链接:http://poj.org/problem?id=2155 Matrix Time Limit: 3000MS   Memory Limit: 65536K Total Submissio ...

  5. POJ 2155 Matrix(二维树状数组+区间更新单点求和)

    题意:给你一个n*n的全0矩阵,每次有两个操作: C x1 y1 x2 y2:将(x1,y1)到(x2,y2)的矩阵全部值求反 Q x y:求出(x,y)位置的值 树状数组标准是求单点更新区间求和,但 ...

  6. POJ 2155 Matrix (二维树状数组)

    Matrix Time Limit: 3000MS   Memory Limit: 65536K Total Submissions: 17224   Accepted: 6460 Descripti ...

  7. POJ 2155 Matrix 【二维树状数组】(二维单点查询经典题)

    <题目链接> 题目大意: 给出一个初始值全为0的矩阵,对其进行两个操作. 1.给出一个子矩阵的左上角和右上角坐标,这两个坐标所代表的矩阵内0变成1,1变成0. 2.查询某个坐标的点的值. ...

  8. POJ 2155 Matrix (二维树状数组)题解

    思路: 没想到二维树状数组和一维的比只差了一行,update单点更新,query求和 这里的函数用法和平时不一样,query直接算出来就是某点的值,怎么做到的呢? 我们在更新的时候不止更新一个点,而是 ...

  9. POJ 2155:Matrix 二维树状数组

    Matrix Time Limit: 3000MS   Memory Limit: 65536K Total Submissions: 21757   Accepted: 8141 Descripti ...

  10. POJ 2155 Matrix(二维树状数组)

    与以往不同的是,这个树状数组是二维的,仅此而已 #include <iostream> #include <cstdio> #include <cstring> # ...

随机推荐

  1. django缓存

    由于Django是动态网站,所有每次请求均会去数据进行相应的操作,当程序访问量大时,耗时必然会更加明显,最简单解决方式是使用:缓存,缓存将一个某个views的返回值保存至内存或者memcache中,5 ...

  2. Union-Find Algorithm

    Union-Find Algrithm is used to check whether two components are connected or not. Examples: By using ...

  3. React 源码解读参考,理解原理。

    Rubix - ReactJS Powered Admin Template 文档:   http://rubix-docs.sketchpixy.com/ ===================== ...

  4. 【转】ArrayList其实就那么一回事儿之源码浅析

    转自:http://www.cnblogs.com/dongying/p/4013271.html?utm_source=tuicool&utm_medium=referral ArrayLi ...

  5. Android SDK打包

    2015年6月18日 14:38:49 星期四 eclipse: 1. 将写好的代码上传版本库 2. 删除 /bin/* 3. eclipse->project->clean... 4. ...

  6. PHP-FPM的常用操作

    PHP-FPM安装完毕之后,没有自带的结束命令,可以通过以下方法结束: 1.首先查看PHP-FPM进程号: ps -ef | grep php-fpm 可以看到master进程号为91790,有两个子 ...

  7. 50道java算法题(一)

    [程序1] 题目:古典问题:有一对兔子,从出生后第3个月起每个月都生一对兔子,小兔子长到第三个月后每个月又生一对兔子,假如兔子都不死,问每个月的兔子总数为多少? 1.程序分析:   兔子的规律为数列1 ...

  8. ACM/ICPC 之 Prim范例(ZOJ1586-POJ1789(ZOJ2158))

    两道Prim解法范例题型,简单的裸Prim,且两题相较以边为重心的Kruskal解法而言更适合以点为重心扩展的Prim解法. ZOJ1586-QS Network 题意:见Code 题解:直接的MST ...

  9. Thread.Sleep in WinRT

    Thread.Sleep in WinRT static void Sleep(int ms) {    new System.Threading.ManualResetEvent(false).Wa ...

  10. WebLogic部署

    1.抓取解压WAR包,放在相应目录下 2.登录部署,激活 http://jingyan.baidu.com/article/c74d6000650d470f6b595d72.html Linux环境中 ...