上一篇我们对树状数组进行了一些分析(详见树状数组学习系列1 之 初步分析——czyuan原创http://hi.baidu.com/czyuan_acm/blog/item/49f02acb487f06f452664fbc.html),这篇主要是对各大OJ有关树状数组的题目进行汇总。
先提个注意点,由于Lowbit(0) = 0,这会导致x递增的那条路径发生死循环,所有当树状数组中可能出现0时,我们都全部加一,这样可以避免0带来的麻烦~~
简单:
POJ 2299 Ultra-QuickSort
http://acm.pku.edu.cn/JudgeOnline/problem?id=2299
求逆序数,可以用经典的归并排序做,也是基本的树状数组题目。
POJ 2352 Stars
http://acm.pku.edu.cn/JudgeOnline/problem?id=2352
题目意思就是求每个星星左下方的星星的个数,由于y轴已经排序好了,我们可以直接用按x轴建立一维树状数组,然后求相当于它前面比它小的个数,模板直接一套就搞定了~~
POJ 1195 Mobile phones
http://acm.pku.edu.cn/JudgeOnline/problem?id=1195
二维的树状数组,直接把Update()和Getsum()改为二维即可。
如Update()函数改为:
void Update(int x, int y, int d)
{ // 注意:当i = 0,0 + Lowbit(0) = 0,会造成死循环!
int i, j;
for (i = x; i < maxn; i += Lowbit(i)) // 注意这里是maxn,是tree[]的大小.
{
for (j = y; j < maxn; j += Lowbit(j))
{
tree[i][j] += d;
}
}
}
POJ 2481 Cows
http://acm.pku.edu.cn/JudgeOnline/problem?id=2481
将E从打到小排序,如果E相等按S排序,然后就跟POJ 2352 Stars做法一样了~~
POJ 3067 Japan
http://acm.pku.edu.cn/JudgeOnline/problem?id=3067
先按第一个坐标排序从大到小排序,如果相等按第二个坐标从大到小排序,然后就又是跟Cows和Stars做法相同了...
POJ 2029 Get Many Persimmon Trees
http://acm.pku.edu.cn/JudgeOnline/problem?id=2029
O(n ^ 2)枚举起点,再用二维树状数组求其中的点数即可。
HOJ 2275 Number sequence
http://202.118.224.210/judge/show.php?Contestid=0&Proid=2275
两个一维树状数组,分别记录在它左边比它小的和在它右边比它大的即可~~
HOJ 1867 经理的烦恼
http://202.118.224.210/judge/show.php?Contestid=0&Proid=1867
先筛法求素数,然后如果从非素数改变成素数就Update(x, 1),如果从素数改变成非素数就Update(x, -1)即可。
Sgu 180 Inversions
http://acm.sgu.ru/problem.php?contest=0&problem=180
经典树状数组 + 离散化。注意结果要用long long~~
SPOJ 1029 Matrix Summation
https://www.spoj.pl/problems/MATSUM/
基本的二维树状数组...
中等:
POJ 2155 Matrix
http://acm.pku.edu.cn/JudgeOnline/problem?id=2155
经典树状数组题目,分析见前一篇文章(树状数组学习系列1 之 初步分析——czyuan原创)~~
POJ 3321 Apple Tree
http://acm.pku.edu.cn/JudgeOnline/problem?id=3321
这题的难点不在于树状数组,而是如果将整棵树映射到数组中。我们可以用DFS()改时间戳的方法,用begin[i]表示以i为根的子树遍历的第一个点,end[i]表示以i为根的子树遍历的最后一个点。
比如数据为:
5
1 2
2 5
2 4
1 3
那么begin[] = {1, 2, 5, 4, 3}, end[] = {5, 4, 5, 4, 3},下标从1开始。
对于每个点都对应一个区间(begin[i], end[i]),如果要改变点a的状态,只要Update(begin[a]),要求该子树的苹果树,即Getsum(begin[a] ) - Getsum(end[a] + 1),(注:这里求和是求x递增的路径的和。)
POJ 1990 MooFest
http://acm.pku.edu.cn/JudgeOnline/problem?id=1990
这题的难点是要用两个一维的树状数组,分别记录在它前面横坐标比它小的牛的个数,和在它前面横坐标比它小的牛的横坐标之和。
按音量排个序,那么式子为:
ans += 1LL * cow[i].volumn * (count * x - pre + total - pre - (i - count) * x);
cow[i].volumn为该牛的能够听到的音量。
count为在第i只牛前面横坐标比它小的牛的个数。
pre为在第i只牛前面横坐标比它小的牛的横坐标之和。
total 表示前i - 1个点的x坐标之和。
分为横坐标比它小和横坐标比它大的两部分计算即可。
Hdu 3015 Disharmony Trees
http://acm.hdu.edu.cn/showproblem.php?pid=3015
跟上题方法相同,只要按它的要求离散化后,按高度降序排序,套用上题二个树状数组的方法即可。
HOJ 2430 Counting the algorithms
http://202.118.224.210/judge/show.php?Contestid=0&Proid=2430
这题其实是个贪心,从左往右或者从右往左,找与它相同的删去即可。先扫描一遍记录第一次出现和第二次出现的位置,然后我们从右到佐,每删去一对,只需要更改左边的位置的树状数组即可,因为右边的不会再用到了。
tju 3243 Blocked Road
http://acm.tju.edu.cn/toj/showp3243.html
这题主要在于如果判断是否连通,我们可以先用j = Getsum(b) – Getsum(a – 1),如果j等于(b – a)或者Getsum(n) – j等于(n – (b – a)),那么点a, b联通。
SPOJ 227 Ordering the Soldiers
http://www.spoj.pl/problems/ORDERS/
这题与正常的树状数组题目正好想反,给定数组b[i]表示i前面比a[i]小的点的个数,求a[]数组。
我们可以先想想朴素的做法,比如b[] = {0, 1, 2, 0, 1},我们用数组c[i]表示还存在的小于等于i的个数,一开始c[] = {1, 2, 3, 4, 5},下标从1开始。
我们从右向左扫描b[]数组,b[5] = 1,说明该点的数是剩下的数中第4大的,也就是小于等于它的有4个,即我们要找最小的j符合c[j] = 4(这里可以想想为什么是最小的,不是最大的,挺好理解的),而c[]是有序的,所以可以用二分来找j,复杂度为O(logn),但现在问题是每次更新c[]要O(n)的复杂度,这里我们就想到树状数组,c[i]表示还存在的小于等于i的个数,这不正好是树状数组的看家本领吗~~所以处理每个位置的复杂度为O(logn * logn),总的复杂度为O(n * logn * logn)。
hdu 2852 KiKi's K-Number
http://acm.hdu.edu.cn/showproblem.php?pid=2852
这题与上面那题类似,只是要求比a大的第k大的数,那我们用Getsum(a)求出小于等于a的个数,那么就是要我们求第k + Getsum(a)大的数,而删除操作只要判断Getsum(a) – Getsum(a - 1)是否为0,为0则说明a不存在。
难题:
POJ 2464 Brownie Points II
http://acm.pku.edu.cn/JudgeOnline/problem?id=2464
这道题用二分也可以做的,这里介绍下树状数组的做法。首先有n个点,过每个点可以做x,y轴,把平面切成BL, TL, TR, BR四个部分,我们现在的问题是如果快速的计算这四个部分的点的个数。
这样我们可以先预处理,先按y坐标排序,求出每个点正左方和正右方的点的个数LeftPoint[], RightPoint[],复杂度为O(n),同样我们再以x坐标排序,求出每个点正上方和正下方点的个数UpPoint[], DownPoint[]。还要求出比点i y坐标大的点的个数 LageY[]。注意:这里要进行下标映射,因为两次排序点的下标是不相同的。
然后按x坐标从小到大排序,x坐标相等则y坐标从小到大排序。我们可以把y坐标放在一个树状数组中。
对于第i个点,求出Getsum(y[i])即为BL的个数,然后Update(y[i])。由于现在是第i点,说明前面有i – 1个点, 那么
TL = i - 1 - LeftPoint[i]- BL;
TR = LargeY[i] - TL – UpPoint[i] ;
BR = n - BL - TL - TR - LeftPoint[i] - RightPont[i] - UpPoint[i] – DownPont[i] - 1;
这样我们就求出四个部分的点的个数,然后判断有没有当前解优,有的话就更新即可~~
UVA 11610 Reverse Prime
http://uva.onlinejudge.org/index.php?option=com_onlinejudge&Itemid=8&page=show_problem&category=78&problem=2657&mosmsg=Submission+received+with+ID+7313177
一道很综合的树状数组题,用到了树状数组中的很多知识点,包括离散化,二分查找等。
1. 先按题目要求筛法素数,找到所有的Reverse Prime。
2. 将这些Reverse Prime离散化,只有78500个左右。树状数组中tree[i]记录比i小的点的个数。
当执行q a操作时,二分查找最小的j, 使得Getsum(j) 等于 ++a(因为a可能为0,所以统一加一)。这步与上面SPOJ 227 Ordering the Soldiers 类似。
3. 当执行d a操作时,先找到a离散化后的值b,然后Update(b, -1)即可。
按这样做后,运行时间为: 0.3s多,感觉很诧异,因为都是0.1s以下的。这里特别感谢liuzhe大牛的指点,其实题目中的Reverse Prime是由10^6以下的素数倒置得到的,那么得到的要求是7位,最后一位一定是0,我们可以对每个除以10处理,那么数的范围就减小了10倍,速度就提高了不少。
最后自己又加了点优化,跑了0.056s,排在第3名~~
3 7313367 czyuan 0.056 C++
czyuan原创,转载请注明出处。
2
Poj上的一些题:
1)poj1195 Mobile phones 难度:1
题意:给定n*n矩阵,和几种在线操作,包括对某一点(x,y)值修改,查询一个矩形(x1,y1,x2,y2)的元素和。
分析:用二维树状数组维护即可。
2)poj2299 Ultra-QuickSort 难度:1
题意:给你n个不同的数,求逆序对个数。
分析:有两种方法:归并排序;树状数组。
这里提一下树状数组的方法。倒着处理,每次统计比当前元素值小的已经处理完的元素的个数,所有的累加起来就是答案。
3) poj2352 star 难度:2
题意:按y递增的顺序给出n颗星星的坐标(y相等则x递增),每个星星的等级,等于在它左边且在它下边(包括水平和垂直方向)的星星的数量,求出等级为0到n-1的星星分别有多少个。
分析:由于星星的坐标是按y的递增给出的,那么就只需考虑水平方向就好了,如果给出一个星星的坐标为(a,b),那么它的等级就等于前面已经输入的x坐标在[0,a]区间的星星数量。
4)poj3067 Japan 难度:2
题意:顺序给两组平行的点依次编号1~N和1~M,给定K个线段在两组点之间,求相交(cross)的线段对有多少个,同一个起点或终点不算相交。
分析:设1~n这组点属于A组,1~m属于B组。用树状数组记录当前每个B中的点与多少个已处理完的A中的点相连,按n~1的顺序处理,即倒着做。例如当前处理的A中的为x,与x相关的B中的点有:a,b,c;则将当前B中1~a-1所有点的值的和(每个点的值即为该点与多少个已处理完的A中的点相连)加入答案,为什么呢?当前B中1~a-1所有点的值都是A中大于x的点贡献的;同理,将1~b-1所有点的值的和加入答案,将1~c-1所有点的值的和加入答案。
5)poj2155 matrix 难度:3
题意:一个n*n的01矩阵,和几种动态操作,包括对子矩阵(x1,y1,x2,y2)的所有元素异或,查询某一点(x,y)的元素值。
分析:仅需修改矩阵四个顶点的值,这四个点是:
(x1,y1)、(x1,y2)、(x2,y1)、(x2,y2)。
对于询问(x,y),用树状数组统计(x,y)右下方的元素和mod 2即为答案。
但对于修改的矩阵出现x1=x2或y1=y2时会出现西西,我是这么避免出错的:
将矩阵每个元素拆成四个点(行数*2,列数*2),修改矩阵(x1,y1,x2,y2)则对这四个修改:
(x1*2-1,y1*2-1)、(x1*2-1,y2*2)、 (x2*2-1,y1*2-1)、(x2*2-1,y2*2)。
6)poj3321 apple tree 难度:2
题意:一棵具有n个节点的树,一开始,每个节点上都有一个苹果。现在给出m组动态的操作:(C,i)是摘掉第i个节点上面的苹果(若苹果不存在,则为加上一个苹果),(Q,i)是查询以第i个节点为根的子树有几个苹果(包括第i个节点)。
分析:树状数组。这道题重点怎么建立树到树状数组的映射关系:利用dfs遍历树,对每个节点进行两次编号,第一次搜到第i个节点时的时间戳,为这个节点管辖区间的下限L[i],然后搜这个节点的子节点,最后搜回来后的时间戳,为这个节点管辖区间的上限R[i] (也为这个节点的下标),如下图所示。接下来就是树状数组部分了。
7)POJ 2481 Cows 难度:3
题意:每个奶牛都有一个吃草的范围,对应一段线段 [s, e], 如果某一头a的区间覆盖了另一头b 的区间,就说a 比b 强,求比每头奶牛强的奶牛数。
分析:此题类似于线段覆盖cover那题,此处不讲解,可以参考cover那题的解题报告。
(这题交了n遍,原来poj卡Qsort!!要用随机Qsort)
8)POJ 2029 Get ManyPersimmon Trees 难度:1
题意:给定N*M的矩形和T个点,让你求A*B的子矩形最多能覆盖几个点。
分析:枚举矩阵的右上端点,用树状数组统计该矩阵内点的个数。
9)POJ 1990 MooFest 难度:3
题意:有n头牛,不同的听力值v,当i,j想要通话时,需要max(v(i),v(j))*|dist[i]-dist[j]|的volume,问这n*(n-1)/2对牛总共的volume是多少。
分析:处理两个东西:max和abs。对于max可以Qsort解决:v大的先处理,处理完就删掉。对于abs,可以拆成两部分处理,设当前的牛为i,对于未处理的牛j,
若dist[j] abs(dist[i]-dist[j])=dist[i]-dist[j], 否则(设这种牛有t2个) abs(dist[i]-dist[j])=dist[j]-dist[i]。 故所有j与i的volume值总和为:t1*dist[i]-sigma(dist[j])+sigma(dist[j])-t2*dist[i]. t1与t2可以用树状数组统计,sigma(dist[j])与sigma(dist[j])也可用树状数组统计。 平时考试的一些题: 10)修路(10月15日ldl大牛出的一道题)难度:3 题意:有n个城市和n-1条道路,呈树状分布,其中省会为1号(根)节点。这n-1条路原先是羊肠小道,后来逐渐变为水泥大道,这期间省长小L有多次出访其他n-1个城市。给你m个操作,操作分为两种类型:修建城市x和y之间的道路、询问小L访问城市x时经过了多少条羊肠小道。(初始时所有道路都为羊肠小道) 分析:类似于上一题apple tree,也是利用了映射转化的思想。先对这棵树dfs,求的开始经过每个节点i的时间戳L[i]和最终离开每个节点i的时间戳R[i],将L[i]和r[i]加入到树状数组中,并将l[i]对应的值赋为1,表示i与i的父亲间有一条小道;将r[i]对应的值赋为-1。因此,对于以i为根的子树中的节点统计羊肠小道时能够统计到节点i与i的父亲间的小道,儿对于不是以i为根的节点统计不到i与i的父亲间的小道(因为离开i时又减去了1)。修i与其父亲间的小路时,只需将L[i]的值减1,R[i]的值加1即可(因为要使不是以i为根的节点不受影响)。注意每次统计的答案要减去1,因为树的根节点没有父亲,也就不存在根节点与其父亲间的小路! 11)线段覆盖cover(某日数据结构小考的题目)难度:3 题意:给你n条线段(左端点为l[i],右端点为r[i]),哪条线段覆盖了最多的线段,输出这个最大值。 分析:按左端点递减的顺序处理,先统计右端点小于当前处理线段的右端点的已处理线段数量,统计的结果即为当前线段覆盖的线段的数量,然后将当前线段的右端点加入树状数组中。由于端点值较大,需要离散化。 12)9月24日xqz的模拟题中有一道叫tree的题, 实际上来自Hdu 3015Disharmony Trees(xqz copy的!) 此题同poj1990. hoj上的一些题(现已找不到hoj这些题了): 13)Hoj1867 经理的烦恼 难度:0 一维树状数组+判素数,此处不详细讲。 14)Hoj2430 Counting the algorithms 难度:2 题意:给出2*N的序列,每个数∈[1,N]出现2次 2个数之间的间隔为得分,求得一个得分后会删除这两个数,问最大得分。 分析:贪心+树状数组。从后往前删除数,用树状数组求得sum(x)个数,对于嵌套的、相互独立的,先删除谁都没关系但对于包含关系的,必须先删除外围的,所以从后开始(从前开始也一样)。 15)hoj2275 Number sequence 难度:2 题意:计算序列 a[] 中, 当 i < j < k, a[i] < a[j] >a[k], 这样的子序列个数。