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.[工具]一些常用宏的制作7.宠物相关(猎人、术士) ...7.A 施放猎人印记的同时宠物攻击 ...7.B 切换到宠物正在攻击的目标并施放毒蛇刺击 ...7.C 召回宠物并使用散射 ...7.D 选择目标,开始攻击 ...7.E 喂食 8.判断目标身上是否有特定BUFF ...8.A 猴之守护+物理攻击 ...https://wow.52pk.com/wowcn/20050531/32297.html
4.《魔兽世界》BetaV更新猎人初体验网络游戏新闻宠物都是猎人从野外驯服的.在相距目标30码内既可使用 拥有宠物后快皆栏上面就会出现宠物控制栏,可以对宠物的行为模式进行设定,可设为侵略性,防御性和消极逃避性。跟术士召唤的生物一样。 经常带同一只宠物战斗会增加他们的忠诚度,但是这个忠诚度怎么计算现在不是很了解...而且偶找了半天也没看到诚度在那里显示.....https://news.17173.com/content/2004-8-21/n853_769142.html
5.蜘蛛蜘蛛的种类数目繁多,自然界中蜘蛛有四万多种。这些蜘蛛大致可分为游猎蜘蛛、结网蜘蛛及洞穴蜘蛛三种。第一类会四处觅食,第二类则结网后守株待兔。而人们作为宠物饲养的大多是第三类:洞穴蜘蛛。它们喜欢躲在沙堆或洞里,在洞口结网,网本身没有黏性,纯粹用来感应猎物大小,并加以捕食。 http://www.360doc.com/content/12/1214/08/5516683_253929811.shtml
1.八方旅人2猎人宠物选择推荐八方旅人2猎人宠物怎么选择八方旅人2是一款最近非常受大家关注和热爱的游戏,那么玩家知道猎人宠物怎么选择吗?今天由小编带来的八方旅人2猎人宠物选择推荐,希望能帮助到你们,让我们一起来看看吧。 猎人宠物选择推荐 开局宠物选择: 两只宠物一个是狼,太阳祝福,专精物理。另一个是猫头鹰,月亮祝福,专精魔法。 https://3g.ali213.net/gl/html/1018277.html
2.[宠物相关][猎人相关][杂谈]猎人野生宠物技能资料百科NGA玩家...经典版本中,猎人宠物训练师只会教给你宠物的基础技能:低吼,耐力,自然护甲,法术抗性(奥 火 冰 自然...https://bbs.nga.cn/read.php?tid=18291663&_ff=-152678&page=1
3.魔兽世界怀旧服猎人宠物怎么选择猎人宠物说明网络游戏游戏攻略魔兽世界怀旧服猎人宠物怎么选择?今天给大家带来的就是魔兽世界怀旧服猎人宠物的选择推荐,感兴趣的朋友不妨阅读下文内容,参考一下吧 GPT4.0+Midjourney绘画+国内大模型 会员永久免费使用! 【如果你想靠AI翻身,你先需要一个靠谱的工具!】 潜伏 40集中 CD10秒 ...https://www.jb51.net/gonglue/691096_7.html
4.关于人与动物感人的故事(通用40则)次日,老猎人怀着忐忑不安的心情对那只藏羚羊开膛扒皮,原来在藏羚羊的子宫里,静静地卧着一只小藏羚羊,它已经成形,自然是死了。 这时候,老猎人方才明白为什么那只藏羚羊要弯下笨重的身子给自己下跪,它是在求猎人留下自己的一条命,以保全怀在腹腔中小藏羚羊的生命啊!老猎人在山坡上挖了坑,将那只藏羚羊连同它那没有...https://www.yuwenmi.com/gushi/54780.html
5.它们高中作文(精选80篇)在平平淡淡的学习、工作、生活中,大家都写过作文,肯定对各类作文都很熟悉吧,作文是一种言语活动,具有高度的综合性和创造性。那么你知道一篇好的作文该怎么写吗?以下是小编为大家收集的它们高中作文,希望能够帮助到大家。▼※目录※▼【1】它们高中作文(1~10)【2】它https://www.ruiwen.com/zuowen/gaozhong/1724525.html