2018.1.292018.2.2树状数组学习笔记(带各种应用)大本营

树状数组是大家很熟悉的数据结构了吧!不明白的话自己手动学习。作者要好好复习树状数组了。

ps:点击每道题目名称可跳转至题目网页。

easy(简单模板):

简单单点加法+求和操作。

#include#include#include#include#include#defineLLlonglong#definemaxn500001usingnamespacestd;inlineintread(){intx=0;boolf=1;charch=getchar();while(!isdigit(ch)){if(ch=='-')f=0;ch=getchar();}while(isdigit(ch)){x=x*10+ch-'0';ch=getchar();}if(f)returnx;return0-x;}intn,m;inttree[maxn<<1];intlowbit(intx){returnx&-x;}voidadd(intx,intk){for(inti=x;i<=n;i+=lowbit(i))tree[i]+=k;}intsum(intx){intres=0;for(inti=x;i>0;i-=lowbit(i))res+=tree[i];returnres;}intmain(){n=read(),m=read();inti,x,y,p;for(i=1;i<=n;i++)add(i,read());for(i=1;i<=m;i++){p=read(),x=read(),y=read();if(p==1)add(x,y);elseprintf("%d\n",sum(y)-sum(x-1));}return0;}

这题跟下面一题类似,因此略。

简单区间加法+求和操作。

这题用线段树的做法当然是废话,但树状数组的做法需要应用一点数学知识:差分。

额,以前写过注释了,现在直接贴上来。

medium(小试牛刀):

额,我真想吐槽一下用归并排序写逆序对的常数真是大到爆炸,于是来一发树状数组吧。

思路:将给的n个数分别捆绑上它们在原序列的位置(可以用struct捆),然后将这些数从大到小排序,从大到小(重点!)依次将每个数的值离散化,然后按照原序列从前往后的顺序将每个数离散化后的值插入树状数组的对应位置。

离散化是因为输入的数有可能很大(10^9?),数组开不下那么多位,而最多只有4W个数,因此离散化后就只需要给树状数组开4W位了。

不难证明其正确性——树状数组中对第x位的查询求的是第1-x位的和,由于是按照原序列从前往后的顺序插入值,先插入树状数组的值的位置一定在后插入树状数组的值的位置的前面。如果先插入的数比后插入的数大,那这两个数就能组成逆序对了。

有人可能会问:树状数组查询的是前缀和,也就是之前插入的比当前数小的数的个数,怎么能查询出比当前数大的数的个数呢?

呃,你是不是没明白上面红字标的那句话?“从大到小依次将每个数的值离散化”这句话的实际操作看这个例子(108644的每位不是离散化成43211,而是12344)。

这样原序列就变成了反序的,这时树状数组看似查询的是之前插入的比当前数小的数的个数,实际上之前那些“比当前数小的数”都是在反序离散化后形成的,它们原本的值是比当前数大的。再加上那些数的位置在当前数前面,那些数就都能和当前数组成逆序对了。

因此对于每一个后插入的数,设这个数在原序列是第x位,求一下树状数组中第1~x位的和(简单query操作)即可求出所有位置在这个数(第x位)前面且原数比当前数在原序列中的数大的数的数量了。最后总数量就是原序列的总逆序对数。

另外,上面只是理论可行的做法,实际上逆序对的定义是iaj(不是ai>=aj),而对于每一个插入的数,树状数组的query操作求得是第1~x位的和,算上了之前插入的等于当前这个数的数,为了把它们去掉,实际查询的时候可以让query只求第1~x-1位的和,这样即可避免将两个相同的数算为逆序对。

当然如果您有强迫症,您可以查询第1~x位的和,只不过要多写两行代码把那些相同的数多查询出的答案减掉。

#include#include#includeusingnamespacestd;intn,tot,tree[100010],d[100010];longlongans;structnode{intval,id;booloperator<(constnode&a)const{returnval>a.val;}}a[100010];intlowbit(intx){returnx&-x;}voidadd(into,intx){for(;o<=n;o+=lowbit(o))tree[o]+=x;}intquery(intx){intsum=0;for(;x>0;x-=lowbit(x))sum+=tree[x];returnsum;}intmain(){scanf("%d",&n);for(inti=1;i<=n;i++){scanf("%d",&a[i].val);a[i].id=i;}sort(a+1,a+n+1);//离散化d[a[1].id]=++tot;for(inti=2;i<=n;i++){if(a[i].val!=a[i-1].val)++tot;//要设相同的数在同一位置d[a[i].id]=tot;}for(inti=1;i<=n;i++){add(d[i],1);ans+=query(d[i]-1);}printf("%lld",ans);return0;}

2.弱点

这道题所在的oj要花钱才能注册账号看题……因此我贴题目吧。

一队勇士正在向你进攻,每名勇士都有一个战斗值ai。但是这队勇士却有一个致命弱点,如果存在iaj>ak,则会影响他们整体的战斗力。我们将这样的一组(i,j,k)称为这队勇士的一个弱点。请求出这队勇士的弱点数目。

输入的第一行是一个整数n,表示勇士的数目。

接下来一行包括n个整数,表示每个勇士的战斗值ai。

输出为一行,包含一个整数。表示这队勇士的弱点数目。

410831SampleOutput4HINT对于30%的数据,3<=n<=100

对于100%的数据,3<=n<=1000000

三维逆序对,很明显跟上一题做法相似。那怎么搬运呢?我们可以基于三维逆序对的中间元素考虑。

思路:既然能用树状数组求第x个数能和第1~x-1个数组成多少逆序对,那肯定也能求第x个数能和第x+1~n个数组成多少逆序对,不过由于树状数组只能用来求前缀和,因此我们可以把后者理解为求第x个数能和第x+1~n个数组成多少反向同序对(大家都理解同序对吧?逆序对的反义词,即满足ij且ai

tip:我写代码的时候为了省事,依然做了从大到小和从小到大的离散化(从小到大的离散化),实际上这道题题面标明了ai不会超过10^6,数组开得下,而且每个ai互不相同,这说明离散化的去重效果就没用了,因此完全可以不用离散化,反序嘛直接把1000001-ai的值插入树状数组就行了。

这题明显可以用树状数组打。

思路:

由于每个数可能很大,数组开不下,我们依然把所有数离散化,不过这不是逆序对,要从小到大离散化。树状数组tree[i]表示离散化后的数i为当前第几小数,将对于每个数x,对树状数组第x位进行add操作+1即可,表示它占了一个排名,这样比x大的数都要往后排一名。

比如序列:53791

离散化后:32451

插入第1个数3后,第i位对应的query(i):00111(query(i)操作在逆序对题中已经讲过,表示求树状数组第1~i位的和)

插入第2个数2后,第i位对应的query(i):01222

插入第3个数4后,第i位对应的query(i):01233

插入第4个数5后,第i位对应的query(i):01234

插入第5个数1后,第i位对应的query(i):12345

相信大家都知道初始序列中的数离散化后的排名是不变的,因此离散化后的数第几小,在原序列中对应的数也是第几小。因此这样做肯定是可行的。

还有一个小问题:怎么查询?

相信大家都注意到query(i)序列是单调不下降的,因此可以使用二分快速查找。那查找几呢?根据题目,我们要求前1,3,5,……个数的中位数。假设我们要求前i个数的中位数,根据求和公式,第(i+1)/2小的数就是中位数。所以只要寻找最小的id使query(id)=(i+1)/2就行了。这个位置id表示的其实是原序列一个数离散化后的数,因此我们把它对应回去,输出它原来的数就行咯。

ps:这题方法太多了,有兴趣了解更简单方法的可以看我写的另一篇随笔。

THE END
1.魔兽世界猎人怎么存放宠物游戏问答3、不能选择解散,解散是将宝宝收回去,依然能够再次召唤 https://www.9game.cn/wenda/1248671.html
2.术士和猎人如的宠物如没有天赋点的支持,将立即消失。的英文翻译...海词词典,最权威的学习词典,专业出版术士和猎人如的宠物如没有天赋点的支持,将立即消失。的英文,术士和猎人如的宠物如没有天赋点的支持,将立即消失。翻译,术士和猎人如的宠物如没有天赋点的支持,将立即消失。英语怎么说等详细讲解。海词词典:学习变容易,记忆很深刻http://m.dict.cn/%E6%9C%AF%E5%A3%AB%E5%92%8C%E7%8C%8E%E4%BA%BA%E5%A6%82%E7%9A%84%E5%AE%A0%E7%89%A9%E5%A6%82%E6%B2%A1%E6%9C%89%E5%A4%A9%E8%B5%8B%E7%82%B9%E7%9A%84%E6%94%AF%E6%8C%81%EF%BC%8C%E5%B0%86%E7%AB%8B%E5%8D%B3%E6%B6%88%E5%A4%B1%E3%80%82
3.图文:《魔兽世界》猎人职业官方介绍网络游戏新闻宠物房 你可以把宠物放在宠物房里。宠物房都在酒馆或信箱旁边,你可以购买更多的空间放以存放更多的宠物。如果你想更换宠物并不扔掉原来的宠物,你就需要在宠物房里更换。 宠物技能: 召唤宠物:召唤出你的宠物 野兽驯服:让猎人训练他的宠物学会猎人掌握的各种技能。会有一个菜单显示所有你收集的宠物技能。可以通过训练...https://news.17173.com/content/2004-8-8/n550_466240.html
1.魔兽世界tbc怀旧服必抓什么宠物《魔兽世界》tbc怀旧服猎人必抓宠物推荐 魔兽世界tbc怀旧服快上线了,在TBC中猎人的宠物进行了大幅度改动,其中最重要的就是攻速统一,宝宝就要看技能伤害了,下面就给大家带来魔兽世界tbc怀旧服猎人必抓宠物推荐。 魔兽世界14天游戏时间免费领取地址:点击查看https://m.3dmgame.com/ol/gl/146681.html
2.怪物猎人崛起猎犬和猫带什么好宠物组合使用心得在本作中有猎犬和猫猫可以带在身边,那么怎么选择呢,这里给大家带来了怪物猎人崛起宠物组合使用心得,一起来看下文中介绍吧。 怪物猎人崛起宠物组合使用心得 猫狗 个人觉得有必要细化到特定武器,以轰龙锤来举例,首先这把武器高攻击负会心,短白斩装备技能,出到会心7,弱特3,超心3,。面板会心只有20%。打弱点也就70%...https://gl.ali213.net/html/2022-1/771593.html
3.猎人宠物可以收藏,是不是就账号共享啦?NGA玩家社区前瞻资讯说猎人宠物可以收藏,那是不是账号共享了呀?新起个猎人但又舍不得老角色里的大脚 ...https://bbs.nga.cn/read.php?tid=39842297
4.魔兽世界怀旧服猎人单刷宠物控制技巧LR宝宝控制回指定地点方法魔兽世界怀旧服中猎人单刷需要掌握的技巧有很多,特别是宠物的控制技巧要娴熟,有的新手猎人玩家还不知道怎么控制宝宝去吸引仇恨做到仇恨分离,比如指定宝宝去一个固定的地方,下面就来为大家分享一下宠物的控制技巧。 LR宝宝控制回指定地点方法: 其实这个技巧很简单,就是利用好停留和被动。 https://www.jb51.net/gonglue/699370.html
5.魔兽世界的掠食者一共有多少种这些野兽的攻击速度和技能各有差异。根据分类,猎人宠物可以分为猫科、狼、熊、迅猛龙、野猪、乌龟、蝙蝠...https://ask.zol.com.cn/x/27134967.html
6.关于人与动物感人的故事(通用40则)次日,老猎人怀着忐忑不安的心情对那只藏羚羊开膛扒皮,原来在藏羚羊的子宫里,静静地卧着一只小藏羚羊,它已经成形,自然是死了。 这时候,老猎人方才明白为什么那只藏羚羊要弯下笨重的身子给自己下跪,它是在求猎人留下自己的一条命,以保全怀在腹腔中小藏羚羊的生命啊!老猎人在山坡上挖了坑,将那只藏羚羊连同它那没有...https://www.yuwenmi.com/gushi/54780.html