link

题意:

给一张n个点m条边的无向图,有q个询问,每次询问给出s,t,l,r,问你能否从s走到t,并且初始为人形,结束时必须为狼形,你是人形的时候必须避开$[1,l)$的节点,狼形的时候必须避开$(r,n]$的节点,你只能在$[L,R]$的节点处变身?

$n,q\leq  2\times 10^5,m\leq 4\times 10^5.$

题解:

get技能——Kruskal重构树

重构树是一个类似堆的结构,节点u比它的所有儿子v的权都要来的小/大(在这题中都有用到),可以用并查集建树。

考虑题中s~t有合法路径相当于,s只经过L~N的点能到达的点集和t只经过1~R的点能到达的点集有交。

那么考虑建出Kruskal最大/小重构树,那么两个点集分别对应到两棵树上的一个子树,求出dfs序转化成两个区间。

把每个点对应到一个坐标$[dfn1,dfn2]$(在两棵树中dfs序上的位置),原问题等价于询问一个矩形$x∈[l1,r1],y∈[l2,r2]$中是否有点,离线树状数组即可。

复杂度$\mathcal{O}(n\log n)$。

code:

  1. #include<bits/stdc++.h>
  2. #include "werewolf.h"
  3. #define rep(i,x,y) for (int i=(x);i<=(y);i++)
  4. #define per(i,x,y) for (int i=(x);i>=(y);i--)
  5. #define ll long long
  6. #define VI vector<int>
  7. using namespace std;
  8. const int N=2e5+;
  9. int u,v,n,m,Q,top,bit[N]; VI ans,G[N];
  10. struct node{
  11. int x,y,ty,op,id;
  12. node(){}
  13. node(int x,int y,int ty,int op,int id):x(x),y(y),ty(ty),op(op),id(id){}
  14. }q[N*];
  15. bool cmp(node x,node y){ return x.x<y.x||x.x==y.x&&x.ty<y.ty; }
  16. struct Kruskal_rebuild_tree{
  17. int cnt,head[N],fa[N],clk,in[N],out[N],f[N][];
  18. struct edge{ int to,nxt; }e[N];
  19. void adde(int x,int y){ e[++cnt].to=y; e[cnt].nxt=head[x]; head[x]=cnt; }
  20. int getfa(int x){ return x==fa[x]?x:fa[x]=getfa(fa[x]); }
  21. void dfs(int u,int ty){
  22. rep (i,,) f[u][i]=f[f[u][i-]][i-];
  23. in[u]=++clk;
  24. for (int i=head[u];i;i=e[i].nxt) f[e[i].to][]=u,dfs(e[i].to,ty);
  25. out[u]=clk;
  26. }
  27. void build(int ty){
  28. cnt=; rep (x,,n) fa[x]=x,head[x]=;
  29. if (!ty){
  30. per (x,n,)
  31. for (auto y:G[x])
  32. if (x<y&&getfa(x)!=(y=getfa(y))) adde(x,y),fa[y]=x;
  33. } else{
  34. rep (x,,n)
  35. for (auto y:G[x])
  36. if (x>y&&getfa(x)!=(y=getfa(y))) adde(x,y),fa[y]=x;
  37. }
  38. rep (i,,n) if (getfa(i)==i) dfs(i,ty);
  39. }
  40. int qry(int x,int lim,int ty){
  41. per (i,,)
  42. if (f[x][i]&&(!ty?f[x][i]>=lim:f[x][i]<=lim)) x=f[x][i];
  43. return x;
  44. }
  45. }T[];
  46. void add(int x){ for (;x<=n;x+=x&-x) bit[x]++; }
  47. int qry(int x){ int s=; for (;x;x-=x&-x) s+=bit[x]; return s; }
  48. VI check_validity(int _n,VI x,VI y,VI s,VI t,VI l,VI r){
  49. n=_n; m=x.size(),Q=s.size();
  50. rep (i,,m-) G[++x[i]].push_back(++y[i]),G[y[i]].push_back(x[i]);
  51. T[].build(); T[].build();
  52. rep (i,,n) q[++top]=node(T[].in[i],T[].in[i],,,);
  53. rep (i,,Q-){
  54. ++s[i],++t[i],++l[i],++r[i];
  55. int u=T[].qry(s[i],l[i],),v=T[].qry(t[i],r[i],);
  56. int l1=T[].in[u],r1=T[].out[u],l2=T[].in[v],r2=T[].out[v];
  57. q[++top]=node(r1,r2,,,i);
  58. if (l1>) q[++top]=node(l1-,r2,,-,i);
  59. if (l2>) q[++top]=node(r1,l2-,,-,i);
  60. if (l1>&&l2>) q[++top]=node(l1-,l2-,,,i);
  61. }
  62. sort(q+,q++top,cmp); ans.resize(Q);
  63. rep (i,,top)
  64. if (!q[i].ty) add(q[i].y); else ans[q[i].id]+=qry(q[i].y)*q[i].op;
  65. rep (i,,Q-) ans[i]=!!ans[i];
  66. return ans;
  67. }

uoj407 【IOI2018】狼人的更多相关文章

  1. [IOI2018]狼人

    [IOI2018]狼人 luogu UOJ 对人形和狼形分别建克鲁斯卡尔重构树 每次询问就是对于两棵树dfs序的一个二维数点,主席树维护 #include<bits/stdc++.h> u ...

  2. LOJ.2865.[IOI2018]狼人(Kruskal重构树 主席树)

    LOJ 洛谷 这题不就是Peaks(加强版)或者归程么..这算是\(IOI2018\)撞上\(NOI2018\)的题了? \(Kruskal\)重构树(具体是所有点按从小到大/从大到小的顺序,依次加入 ...

  3. [IOI2018]狼人——kruskal重构树+可持久化线段树

    题目链接: IOI2018werewolf 题目大意:给出一张$n$个点$m$条边的无向图,点和边可重复经过,一个狼人初始为人形,有$q$次询问,每次询问要求人形态只能处于编号不小于$L$的点,狼形态 ...

  4. Luogu4899 IOI2018狼人(kruskal重构树+主席树)

    可以发现询问的即是“由起点开始‘只经过编号大于等于l的点’所形成的连通块”与“由终点开始‘只经过编号小于等于r的点’所形成的连通块”是否有交集.于是建出重构树,就可以知道每个询问的连通情况了.现在要知 ...

  5. ghj1222被坑记录[不持续更新]

    考试注意事项:link1 link2 (密码:wangle) 调不出来bug,可以先透彻一会儿或者是上个厕所或者坐一会别的题(间隔至少20min),然后通读代码 -1. 考试先读题,读题之后搞出一个做 ...

  6. [note]克鲁斯卡尔重构树

    克鲁斯卡尔重构树 又叫并查集重构树 大概在NOI2018之前还是黑科技 现在?烂大街了 主要是针对图上的对边有限制的一类问题 比如每次询问一个点u不能经过边权大于w的边能走到的第k大点权是多少 也就是 ...

  7. NOIWC2019游记

    更新完了? ghj1222这个智障因为NOIP考的太菜没有去THUWC和PKUWC,但是NOIWC还是苟进去了 由于已经结束了,好多事实忘了,所以可能不完整 2019/1/23 Wednesday 明 ...

  8. [IOI2018] werewolf 狼人

    [IOI2018] werewolf 狼人 IOI2018题解 (其实原题强制在线,要用主席树) 代码: 注意: 1.下标从0~n-1 2.kruskal重构树开始有n个节点,tot从n开始,++to ...

  9. [IOI2018] werewolf 狼人 kruskal重构树,主席树

    [IOI2018] werewolf 狼人 LG传送门 kruskal重构树好题. 日常安利博客文章 这题需要搞两棵重构树出来,这两棵重构树和我们平时见过的重构树有点不同(据说叫做点权重构树?),根据 ...

随机推荐

  1. JS三种消息框的使用

    <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...

  2. 配置虚拟机时间使其与国内时间同步,linux时间 ntp

    设置系统时间 [root@node2 ~]# date -s "10/30/18 09:30:00"Tue Oct 30 09:30:00 PDT 2018[root@node2 ...

  3. MeasureSpec介绍及使用详解

    一个MeasureSpec封装了父布局传递给子布局的布局要求,每个MeasureSpec代表了一组宽度和高度的要求.一个MeasureSpec有大小和模式组成.他有三种模式: UNSPECIFIED ...

  4. SQl 跨服务器查询脚本示例

    1.采用OPENDATASOURCE select top 10 *from OPENDATASOURCE('SQLOLEDB','Data Source=IP地址;User ID=连接用户名称;Pa ...

  5. Java关于网络编程回顾

    一.Java网络编程三要素:1.IP地址:是要确定发送的地址,IP地址一般分为5类. 2.端口:要确定发送的程序是哪一个,端口的范围是0--65535,其中0-1024是系统使用或保留端口 3.协议: ...

  6. 01 Getting Started 开始

    Getting Started 开始 Install the Go tools Test your installation Uninstalling Go Getting help   Downlo ...

  7. eclipse各种报错

    1.控制台报这个错是由于tomcat的session缓存的问题; org.apache.catalina.session.StandardManager doLoad 造成原因:上次未正确关闭tomc ...

  8. Tomcat 上传war包后 会自动部署

  9. HDU 4597 Play Game(区间DP(记忆化搜索))

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4597 题目大意: 有两行卡片,每个卡片都有各自的权值. 两个人轮流取卡片,每次只能从任一行的左端或右端 ...

  10. sql server 2000系统表sysproperties在SQL 2008中无效的问题

    Sqlserver有一个扩展属性系统表sysproperties,因为只接触过MSSQL2005及以后的版本,在生产库2008版本及联机文档上搜了下都找不到这个系统表,后来发现这个系统表在2005版本 ...