题意

考虑整体二分。

考虑路径\((x,y)\)被路径\((u,v)\)包含需要满足什么条件:

设\(dfn_x\)表示\(x\)的\(dfs\)序,\(low_x=dfn_x+size_x-1\),即子树最后一个的\(dfs\)序。

我们钦定\(dfn_x<dfn_y,dfn_u<dfn_v\)

1.\(lca(x,y)!=x\)

\(u\)需要在\(x\)的子树中,\(v\)需要在\(y\)的子树中,体现在\(dfs\)序上就是:

\(dfn_x\leqslant dfn_u\leqslant low_x,dfn_y\leqslant dfn_v\leqslant low_y\)

即一个左上角为\((dfn_x,low_x)\),右下角为\((dfn_y,low_y)\)的矩形内所有\((dfn_u,dfn_v)\)都可以覆盖\((x,y)\)。

2.\(lca(x,y)=x\)

这时\(v\)需要在\(y\)的子树中,而\(u\)需要不在\(x\)包含\(y\)的那颗子树中,我们设\(z\)为\(x\)的儿子中子树包含\(y\)的那个,类比上面,我们能得到(注意\(dfn_u<dfn_v\)):

\(1\leqslant dfn_u\leqslant dfn_z-1,dfn_y\leqslant dfn_v\leqslant low_y||low_x+1\leqslant dfn_v\leqslant n,dfn_y\leqslant dfn_u\leqslant low_y\)

于是查一个点被多少路径包含即查询一个矩形内的点数,显然是扫描线。

code:

  1. #include<bits/stdc++.h>
  2. using namespace std;
  3. #define re register
  4. const int maxn=40010;
  5. int n,m,Q,cnt,tim,tot,num;
  6. int head[maxn],dfn[maxn],low[maxn],size[maxn],top[maxn],pre[maxn],dep[maxn],son[maxn],a[maxn],ans[maxn];
  7. struct edge{int to,nxt;}e[maxn<<1];
  8. struct Query{int op,x,y1,y2,k,id;}qr[maxn<<2],tmpql[maxn<<2],tmpqr[maxn<<2];
  9. inline int read()
  10. {
  11. char c=getchar();int res=0,f=1;
  12. while(c<'0'||c>'9'){if(c=='-')f=-1;c=getchar();}
  13. while(c>='0'&&c<='9')res=res*10+c-'0',c=getchar();
  14. return res*f;
  15. }
  16. inline bool cmp(Query a,Query b){return a.x==b.x?a.op<b.op:a.x<b.x;}
  17. inline void add(int u,int v)
  18. {
  19. e[++cnt].nxt=head[u];
  20. head[u]=cnt;
  21. e[cnt].to=v;
  22. }
  23. void dfs1(int x,int fa)
  24. {
  25. size[x]=1;dep[x]=dep[fa]+1;pre[x]=fa;
  26. for(int i=head[x];i;i=e[i].nxt)
  27. {
  28. int y=e[i].to;
  29. if(y==fa)continue;
  30. dfs1(y,x);size[x]+=size[y];
  31. if(size[y]>size[son[x]])son[x]=y;
  32. }
  33. }
  34. void dfs2(int x,int tp)
  35. {
  36. top[x]=tp;dfn[x]=++tim;
  37. if(son[x])dfs2(son[x],tp);
  38. for(int i=head[x];i;i=e[i].nxt)
  39. {
  40. int y=e[i].to;
  41. if(y==son[x]||y==pre[x])continue;
  42. dfs2(y,y);
  43. }
  44. low[x]=tim;
  45. }
  46. inline int lca(int x,int y)
  47. {
  48. while(top[x]!=top[y])
  49. {
  50. if(dep[top[x]]<dep[top[y]])swap(x,y);
  51. x=pre[top[x]];
  52. }
  53. return dep[x]<dep[y]?x:y;
  54. }
  55. inline int jump(int x,int goal)
  56. {
  57. while(top[x]!=top[goal])
  58. {
  59. if(pre[top[x]]==goal)return top[x];
  60. x=pre[top[x]];
  61. }
  62. return son[goal];
  63. }
  64. struct Tree_arry
  65. {
  66. #define lowbit(x) (x&-x)
  67. int a[maxn];
  68. inline void add(int x,int k){for(int i=x;i<=n;i+=lowbit(i))a[i]+=k;}
  69. inline int query(int x){int res=0;for(int i=x;i;i-=lowbit(i))res+=a[i];return res;}
  70. }tr;
  71. void solve(int L,int R,int l,int r)
  72. {
  73. if(L>R)return;
  74. if(l==r)
  75. {
  76. for(int i=L;i<=R;i++)if(qr[i].op==2)ans[qr[i].id]=l;
  77. return;
  78. }
  79. int mid=(l+r)>>1,cntl=0,cntr=0;
  80. for(re int i=L;i<=R;i++)
  81. {
  82. //cerr<<"!"<<' ';
  83. if(qr[i].op==2)
  84. {
  85. int tmp=tr.query(qr[i].y1);
  86. if(tmp>=qr[i].k)tmpql[++cntl]=qr[i];
  87. else qr[i].k-=tmp,tmpqr[++cntr]=qr[i];
  88. }
  89. else
  90. {
  91. //cerr<<qr[i].y1<<' '<<qr[i].y2+1<<endl;
  92. if(qr[i].k<=mid)tr.add(qr[i].y1,qr[i].op),tr.add(qr[i].y2+1,-qr[i].op),tmpql[++cntl]=qr[i];
  93. else tmpqr[++cntr]=qr[i];
  94. //cerr<<"end"<<endl;
  95. }
  96. }
  97. for(re int i=1;i<=cntl;i++)qr[L+i-1]=tmpql[i];
  98. for(re int i=1;i<=cntr;i++)qr[L+cntl+i-1]=tmpqr[i];
  99. solve(L,L+cntl-1,l,mid);solve(L+cntl,R,mid+1,r);
  100. }
  101. int main()
  102. {
  103. n=read(),m=read(),Q=read();
  104. for(re int i=1;i<n;i++)
  105. {
  106. int u=read(),v=read();
  107. add(u,v),add(v,u);
  108. }
  109. dfs1(1,0),dfs2(1,1);
  110. for(re int i=1;i<=m;i++)
  111. {
  112. int x=read(),y=read(),k=read(),z=lca(x,y);a[++num]=k;
  113. if(dfn[x]>dfn[y])swap(x,y);
  114. if(z==x)
  115. {
  116. int w=jump(y,x);
  117. if(dfn[w]>1)
  118. {
  119. qr[++tot]=(Query){1,1,dfn[y],low[y],k,0};
  120. qr[++tot]=(Query){-1,dfn[w],dfn[y],low[y],k,0};
  121. }
  122. if(low[w]<n)
  123. {
  124. qr[++tot]=(Query){1,dfn[y],low[w]+1,n,k,0};
  125. qr[++tot]=(Query){-1,low[y]+1,low[w]+1,n,k,0};
  126. }
  127. }
  128. else
  129. {
  130. qr[++tot]=(Query){1,dfn[x],dfn[y],low[y],k,0};
  131. qr[++tot]=(Query){-1,low[x]+1,dfn[y],low[y],k,0};
  132. }
  133. }
  134. sort(a+1,a+num+1);num=unique(a+1,a+num+1)-(a+1);
  135. for(re int i=1;i<=tot;i++)qr[i].k=lower_bound(a+1,a+num+1,qr[i].k)-a;
  136. for(re int i=1;i<=Q;i++)
  137. {
  138. int x=read(),y=read(),k=read();
  139. if(dfn[x]>dfn[y])swap(x,y);
  140. qr[++tot]=(Query){2,dfn[x],dfn[y],0,k,i};
  141. }
  142. sort(qr+1,qr+tot+1,cmp);
  143. solve(1,tot,1,num);
  144. for(re int i=1;i<=Q;i++)printf("%d\n",a[ans[i]]);
  145. return 0;
  146. }

luoguP3242 [HNOI2015]接水果的更多相关文章

  1. BZOJ4009: [HNOI2015]接水果

    4009: [HNOI2015]接水果 Description 风见幽香非常喜欢玩一个叫做 osu!的游戏,其中她最喜欢玩的模式就是接水果. 由于她已经DT FC 了The big black,  她 ...

  2. BZOJ 4009: [HNOI2015]接水果

    4009: [HNOI2015]接水果 Time Limit: 60 Sec  Memory Limit: 512 MBSubmit: 636  Solved: 300[Submit][Status] ...

  3. [BZOJ4009][HNOI2015]接水果(整体二分)

    [HNOI2015]接水果 时间限制:60s      空间限制:512MB 题目描述 风见幽香非常喜欢玩一个叫做 osu!的游戏,其中她最喜欢玩的模式就是接水果. 由于她已经DT FC 了The b ...

  4. 洛谷 P3242 [HNOI2015]接水果 解题报告

    P3242 [HNOI2015]接水果 题目描述 风见幽香非常喜欢玩一个叫做 \(osu!\) 的游戏,其中她最喜欢玩的模式就是接水果.由于她已经\(DT\) \(FC\) 了\(\tt{The\ b ...

  5. [洛谷P3242] [HNOI2015]接水果

    洛谷题目链接:[HNOI2015]接水果 题目描述 风见幽香非常喜欢玩一个叫做 osu!的游戏,其中她最喜欢玩的模式就是接水果.由于她已经DT FC 了The big black, 她觉得这个游戏太简 ...

  6. 【BZOJ4009】[HNOI2015]接水果 DFS序+整体二分+扫描线+树状数组

    [BZOJ4009][HNOI2015]接水果 Description 风见幽香非常喜欢玩一个叫做 osu!的游戏,其中她最喜欢玩的模式就是接水果.由于她已经DT FC 了The big black, ...

  7. [HNOI2015]接水果[整体二分]

    [HNOI2015]接水果 给出一个树上路径集合\(S\) 多次询问\(x,y\)中的\(k\)小值 如果你问我数列上那么我会 树上的话 树上差分了吧直接?- 令 \(st_x<st_y\) 1 ...

  8. [Luogu3242][HNOI2015]接水果

    Luogu 我今天做两道整体二分结果全都是BZOJ权限题??? sol 我们抓住"盘子的路径是水果的路径的子路径"这个条件. 考虑每一个盘子路径\((u,v)\),讨论它可以作为哪 ...

  9. Luogu3242:[HNOI2015]接水果

    题面 Luogu3242 Sol 考虑每个盘子怎样才能接到一个水果 分两种情况: 盘子的\(x, y\)在一条链上,那么水果的两点就要在这条链之外 不在的话,水果的两点就分别在盘子的两点的子树中 记录 ...

随机推荐

  1. AcWing 77. 翻转单词顺序

    习题地址 https://www.acwing.com/problem/content/description/73/ 题目描述输入一个英文句子,翻转句子中单词的顺序,但单词内字符的顺序不变. 为简单 ...

  2. CF1248F Catowice City

    题目链接 problem 有\(n\)个人,每个人家有一只猫.每个人都认识一些猫(其中肯定包括自己家的猫).选出\(j\)个人和\(k\)只猫\((j,k\ge 1)\).使得\(j+k=n\)且选出 ...

  3. 【2019.8.15 慈溪模拟赛 T1】插头(plugin)(二分+贪心)

    二分 首先,可以发现,最后的答案显然满足可二分性,因此我们可以二分答案. 然后,我们只要贪心,就可以验证了. 贪心 不难发现,肯定会优先选择能提供更多插座的排插,且在确定充电器个数的情况下,肯定选择能 ...

  4. 并发编程实战之并发下的socket套接字编程

    目录 一.python单线程下实现多个socket并发 1.1 服务端 1.2 客户端 一.python单线程下实现多个socket并发 1.1 服务端 import sys # import soc ...

  5. electron内监控目标网站cookie的变化,查找指定的cookie

    let cookieInstance = win.webContents.session.cookies; cookieInstance.on('changed', (e, cookie, cause ...

  6. php strlen和mb_strlen

    结果: 结论:如果没有中文,尽量使用strlen

  7. 记一次token安全认证的实践

    阅读此文前请先阅读上一篇SpringBoot整合JWT实现用户认证了解JWT. 背景介绍: 因项目需求,有PC端 APP端和小程序端,但登陆接口是同一个,然而微服务也无法使用传统的session解决用 ...

  8. 3 JDK并发包

    JDK内部提供了大量实用的API和框架.本章主要介绍这些JDK内部功能,主要分为3大部分: 首先,介绍有关同步控制的工具,之前介绍的synchronized就是一种同步控制手段,将介绍更加丰富的多线程 ...

  9. .net core的服务器模式和工作站模式

    来源:济南小老虎 .NET Core是一个开源通用的开发框架,具有跨平台能力,我们在享受其性能飙升的同时,也面临了一些问题.通过观察 NetCore 程序的线上运行情况发现 ,负载高的情况下应用程序占 ...

  10. python基础(27):类成员的修饰符、类的特殊成员

    1. 类成员的修饰符 类的所有成员在上一步骤中已经做了详细的介绍,对于每一个类的成员而言都有两种形式: 公有成员,在任何地方都能访问 私有成员,只有在类的内部才能方法 私有成员和公有成员的定义不同:私 ...