题意

给出一棵以1为根节点树,求每个节点的子树中到该节点距离<=l的节点的个数

题解

方法1:倍增+差分数组

首先可以很容易的转化问题,考虑每个节点对哪些节点有贡献

即每次对于一个节点,找到其第l个父亲,这个操作可以用倍增在logn时间内完成

找到后将x-y这一段区间都加1,很容易想到用差分数组维护

方法2:主席树

考虑节点x和节点x的子树中的一个节点y,记点x到根节点的距离为dis[x]

若dis[y]-dis[x]<=l则满足条件

将不等式变形可得dis[y]<=dis[x]+l

即对每个点以dis[ ]为权值插入树中,查询时查找比dis[x]+l小的节点个数

这时很容易想到主席树,第一维对dis[x](维护的前缀也就是查询的答案),第二维对节点编号

又由于先决条件是节点y是在x的子树中的,所以我们应弄出一个dfs序

对于每一个节点,其子树即num[x]-----子树中num[y]的最大值

其次,插入时为避免对之后的节点造成影响,应将dis[]从小到大进行插入

注意到dis数据范围较大考虑对其离散化

还有就是,主席树的题目空间往往不能浪费太多

方法3:左偏树

可以考虑,对于节点x的子树中的一个节点

若x与这个节点的距离>l,那么x的父亲与这个节点的距离也一定>l

所以可以考虑用左偏树来维护

对于每个节点x,枚举它的儿子y,维护一个大根堆

另外,两个点之间的距离可以用dis[1,y]-dis[1,x]表示

**没仔细看题数据是longlong啊。。

代码

方法1:倍增+差分数组

方法2:主席树

  1. uses math;
  2. type re=record
  3. a,b,c,num:int64;
  4. end;
  5. type ree=record
  6. h,t,x:longint;
  7. end;
  8. var
  9. i,j:longint;
  10. now,m,n,c,d,k,l,o,ans,x,tmp:int64;
  11. a,dis:array[..]of re;
  12. f:array[..]of boolean;
  13. num,fa,head,q:array[..]of int64;
  14. p:array[..]of ree;
  15. procedure arr(x,y,z:int64);
  16. begin
  17. inc(l);
  18. a[l].a:=head[x];
  19. a[l].b:=y;
  20. a[l].c:=z;
  21. head[x]:=l;
  22. end;
  23. function dfs(x,y:int64):int64;
  24. var u,v:longint;
  25. begin
  26. f[x]:=false; dis[x].a:=y; dis[x].b:=x;
  27. inc(now); dis[x].num:=now;
  28. u:=head[x]; dfs:=now;
  29. while u<> do
  30. begin
  31. v:=a[u].b;
  32. if f[v] then dfs:=max(dfs,dfs(v,y+a[u].c));
  33. u:=a[u].a;
  34. end;
  35. dis[x].c:=dfs;
  36. end;
  37. procedure swap(var x,y:re);
  38. var tmp:re;
  39. begin
  40. tmp:=x; x:=y; y:=tmp;
  41. end;
  42. procedure qsort(h,t:int64);
  43. var i,j,mid:int64;
  44. begin
  45. i:=h; j:=t; mid:=dis[(h+t) div ].a;
  46. repeat
  47. while dis[i].a<mid do inc(i);
  48. while dis[j].a>mid do dec(j);
  49. if i<=j then
  50. begin
  51. swap(dis[i],dis[j]);
  52. inc(i); dec(j);
  53. end;
  54. until i>j;
  55. if i<t then qsort(i,t);
  56. if h<j then qsort(h,j);
  57. end;
  58. procedure build(x,h,t:int64);
  59. var mid:int64;
  60. begin
  61. p[x].h:=x*; p[x].t:=x*+; now:=max(now,x*+);
  62. if h=t then exit;
  63. mid:=(h+t) div ;
  64. build(x*,h,mid); build(x*+,mid+,t);
  65. end;
  66. function find(x:int64):int64;
  67. var h,t,mid:int64;
  68. begin
  69. h:=; t:=n;
  70. while h<t do
  71. begin
  72. mid:=(h+t) div +;
  73. if dis[mid].a<=x then h:=mid else t:=mid-;
  74. end;
  75. exit(h);
  76. end;
  77. procedure insert(pre,x,h,t:int64);
  78. var tmp,mid:longint;
  79. begin
  80. inc(now); p[now]:=p[pre]; inc(p[now].x); tmp:=now;
  81. if h=t then exit;
  82. mid:=(h+t) div ;
  83. if (x<=mid) then
  84. begin
  85. insert(p[now].h,x,h,mid);
  86. p[tmp].h:=tmp+;
  87. end else
  88. begin
  89. insert(p[now].t,x,mid+,t);
  90. p[tmp].t:=tmp+;
  91. end;
  92. end;
  93. function query(x,h1,t1,h,t:int64):int64;
  94. var mid:int64;
  95. begin
  96. if (h>t1) or (t<h1) then exit();
  97. if (h<=h1) and (t1<=t) then exit(p[x].x);
  98. mid:=(h1+t1) div ;
  99. exit(query(p[x].h,h1,mid,h,t)+query(p[x].t,mid+,t1,h,t));
  100. end;
  101. begin
  102. readln(n,k); fillchar(f,sizeof(f),true);
  103. for i:= to n- do
  104. begin
  105. read(c,d); arr(i+,c,d); arr(c,i+,d);
  106. end;
  107. dfs(,);
  108. qsort(,n); o:=;
  109. for i:= to n do
  110. begin
  111. if (i=) or (dis[i].a<>dis[i-].a) then
  112. inc(o);
  113. num[i]:=o;
  114. end;
  115. now:=; build(,,n);
  116. fa[]:=;
  117. for i:= to n do
  118. begin
  119. if (i<>) and (num[i]=num[i-]) then
  120. begin
  121. tmp:=now;
  122. insert(fa[num[i]],dis[i].num,,n);
  123. fa[num[i]]:=tmp+;
  124. end
  125. else
  126. begin
  127. fa[num[i]]:=now+;
  128. insert(fa[num[i]-],dis[i].num,,n);
  129. end;
  130. end;
  131. for i:= to n do
  132. begin
  133. x:=find(dis[i].a+k);
  134. ans:=query(fa[num[x]],,n,dis[i].num,dis[i].c);
  135. q[dis[i].b]:=ans;
  136. end;
  137. for i:= to n do writeln(q[i]);
  138. end.

方法3:左偏树

  1. type re=record
  2. a,b,c:int64;
  3. end;
  4. var
  5. i,j:longint;
  6. m,n,ans,l,c,d:int64;
  7. k:int64;
  8. f:array[..]of boolean;
  9. a:array[..]of re;
  10. dis,left1,right1,fa,cnt,ll,head,vv,num:array[..]of int64;
  11. procedure arr(x,y,z:int64);
  12. begin
  13. inc(l);
  14. a[l].a:=head[x];
  15. a[l].b:=y;
  16. a[l].c:=z;
  17. head[x]:=l;
  18. end;
  19. function getfa(x:int64):int64;
  20. begin
  21. while (fa[x]<>) do x:=fa[x];
  22. exit(x);
  23. end;
  24. procedure swap(var x,y:int64);
  25. var tmp:int64;
  26. begin
  27. tmp:=x; x:=y; y:=tmp;
  28. end;
  29. function merge(x,y:int64):int64;
  30. var tmp:int64;
  31. begin
  32. if (x=) or (y=) then exit(x+y);
  33. if (dis[x]<dis[y]) then swap(x,y);
  34. right1[x]:=merge(right1[x],y);
  35. fa[right1[x]]:=x;
  36. if (ll[left1[x]]<ll[right1[x]]) then swap(left1[x],right1[x]);
  37. ll[x]:=ll[right1[x]]+; cnt[x]:=cnt[left1[x]]+cnt[right1[x]]+;
  38. exit(x);
  39. end;
  40. function delete(x:int64):int64;
  41. var tmp:int64;
  42. begin
  43. fa[left1[x]]:=; fa[right1[x]]:=;
  44. if left1[x]<> then tmp:=left1[x] else tmp:=right1[x];
  45. merge(left1[x],right1[x]);
  46. exit(getfa(tmp));
  47. end;
  48. procedure dfs(x,y:int64);
  49. var u,v,c,d,goal,ans,z:int64;
  50. begin
  51. ans:=; f[x]:=false; dis[x]:=y;
  52. u:=head[x];
  53. while u<> do
  54. begin
  55. v:=a[u].b;
  56. if f[v] then
  57. begin
  58. dfs(v,y+a[u].c);
  59. c:=getfa(v); goal:=y+k;
  60. while (c<>) and (dis[c]>goal) do
  61. begin
  62. c:=delete(c);
  63. end;
  64. if c<> then
  65. begin
  66. ans:=ans+cnt[c];
  67. z:=getfa(x);
  68. merge(z,c);
  69. end;
  70. end;
  71. u:=a[u].a;
  72. end;
  73. if k>= then vv[x]:=ans+ else vv[x]:=ans;
  74. end;
  75. begin
  76. readln(n,k);
  77. if k< then writeln('');
  78. for i:= to n- do
  79. begin
  80. read(c,d); arr(i+,c,d); arr(c,i+,d);
  81. end;
  82. for i:= to n do cnt[i]:=;
  83. fillchar(f,sizeof(f),true);
  84. dfs(,);
  85. for i:= to n do writeln(vv[i]);
  86. end.

[BZOJ3011][Usaco2012 Dec]Running Away From the Barn的更多相关文章

  1. bzoj3011 [Usaco2012 Dec]Running Away From the Barn 左偏树

    题目传送门 https://lydsy.com/JudgeOnline/problem.php?id=3011 题解 复习一下左偏树板子. 看完题目就知道是左偏树了. 结果这个板子还调了好久. 大概已 ...

  2. 【BZOJ3011】[Usaco2012 Dec]Running Away From the Barn 可并堆

    [BZOJ3011][Usaco2012 Dec]Running Away From the Barn Description It's milking time at Farmer John's f ...

  3. BZOJ 3011: [Usaco2012 Dec]Running Away From the Barn( dfs序 + 主席树 )

    子树操作, dfs序即可.然后计算<=L就直接在可持久化线段树上查询 -------------------------------------------------------------- ...

  4. BZOJ_3011_[Usaco2012 Dec]Running Away From the Barn _可并堆

    BZOJ_3011_[Usaco2012 Dec]Running Away From the Barn _可并堆 Description 给出以1号点为根的一棵有根树,问每个点的子树中与它距离小于l的 ...

  5. [Usaco2012 Dec]Running Away From the Barn

    题目描述 给出以1号点为根的一棵有根树,问每个点的子树中与它距离小于等于l的点有多少个. 输入格式 Line 1: 2 integers, N and L (1 <= N <= 200,0 ...

  6. BZOJ_3012_[Usaco2012 Dec]First!_trie树+拓扑排序

    BZOJ_3012_[Usaco2012 Dec]First!_trie树+拓扑排序 题意: 给定n个总长不超过m的互不相同的字符串,现在你可以任意指定字符之间的大小关系.问有多少个串可能成为字典序最 ...

  7. 【BZOJ3012】[Usaco2012 Dec]First! Trie树+拓补排序

    [BZOJ3012][Usaco2012 Dec]First! Description Bessie has been playing with strings again. She found th ...

  8. [USACO 12DEC]Running Away From the Barn

    Description It's milking time at Farmer John's farm, but the cows have all run away! Farmer John nee ...

  9. USACO Running Away From the Barn /// 可并堆 左偏树维护大顶堆

    题目大意: 给出以1号点为根的一棵有根树,问每个点的子树中与它距离小于等于m的点有多少个 左偏树 https://blog.csdn.net/pengwill97/article/details/82 ...

随机推荐

  1. js模板引擎-art-template常用总结(转)

    原文:https://www.cnblogs.com/shiyou00/p/6841801.html art-template javascript 模板引擎,官网:https://github.co ...

  2. 解决由腾讯qq浏览器引起win10系统桌面图标不停的闪烁问题

    win10系统桌面图标不停的闪烁,虽然不会引起太大问题,但是看着实在郁闷在网上搜索了很久,像停止问题报告服务,重置为默认应用都无解,了解到大概是软件兼容性问题于是打开服务管理器,一个一个关闭不是微软的 ...

  3. Freemaker:操作集合

    <#if (id?index_of('Base') >= 0)> <choose> <when test="rootOrgID !=null and ro ...

  4. Django 笔记(二) 新建 ~ 渲染

    新建APP python manange.py startapp app_name 然后右键 pycharm 的项目目录,将新建的目录从服务器上下载进来 URL(Uniform Resoure Loc ...

  5. Jquery无刷新实时更新表格数据

    html代码: <style> .editbox { display:none } .editbox { font-size:14px; width:70px; background-co ...

  6. Mysql哪些字段适合建立索引

    数据库建立索引常用的规则如下: 1.表的主键.外键必须有索引: 2.数据量超过300的表应该有索引: 3.经常与其他表进行连接的表,在连接字段上应该建立索引: 4.经常出现在Where子句中的字段,特 ...

  7. Nginx的进程模型及高可用方案(OpenResty)

    1. Nginx 进程模型简介 Nginx默认采用多进程工作方式,Nginx启动后,会运行一个master进程和多个worker进程.其中master充当整个进程组与用户的交互接口,同时对进程进行监护 ...

  8. NMT 机器翻译

    本文近期学习NMT相关知识,学习大佬资料,汇总便于后期复习用,有问题,欢迎斧正. 目录 RNN Seq2Seq Attention Seq2Seq + Attention Transformer Tr ...

  9. requests中get和post传参

    get请求 get(url, params=None, **kwargs) requests实现get请求传参的两种方式 方式一: import requests url = 'http://www. ...

  10. Oracle logminer 日志挖掘

    Table of Contents 1. LOGMNR简介 2. 创建数据字典 2.1. 外部文件存储数据字典 2.2. redo log 存储数据字典 3. 添加需要分析的文件 4. 开始分析文件 ...