CD操作

倍增法  https://i.cnblogs.com/EditPosts.aspx?postid=8605845

Time Limit : 10000/5000ms (Java/Other)   Memory Limit : 65535/32768K (Java/Other)
Total Submission(s) : 4   Accepted Submission(s) : 3
Problem Description
  在Windows下我们可以通过cmd运行DOS的部分功能,其中CD是一条很有意思的命令,通过CD操作,我们可以改变当前目录。
  这里我们简化一下问题,假设只有一个根目录,CD操作也只有两种方式:
  
  1.
CD 当前目录名\...\目标目录名 (中间可以包含若干目录,保证目标目录通过绝对路径可达)
  2. CD ..
(返回当前目录的上级目录)
  
  现在给出当前目录和一个目标目录,请问最少需要几次CD操作才能将当前目录变成目标目录?
 
Input
输入数据第一行包含一个整数T(T<=20),表示样例个数;每个样例首先一行是两个整数N和M(1<=N,M<=100000),表示有N个目录和M个询问;接下来N-1行每行两个目录名A
B(目录名是只含有数字或字母,长度小于40的字符串),表示A的父目录是B。最后M行每行两个目录名A
B,表示询问将当前目录从A变成B最少要多少次CD操作。数据保证合法,一定存在一个根目录,每个目录都能从根目录访问到。
 
Output
请输出每次询问的结果,每个查询的输出占一行。
 
Sample Input
2
3 1
B A
C A
B C

3 2
B A
C B
A C
C A

 
Sample Output
2
1
2
 
  1. 从父目录到任意一个子目录的距离都是1
  2. 用一个flag记录一下询问是从左到右还是从右到左,有几种情况 
    • 左边是右边的父亲,距离为1
    • 右边是左边的父亲,距离是dis[v]到lca(u,v)
    • 其他情况+1

其他就是离线求Tarjan的lca了

 
 
 
  1. #include<iostream>
  2. #include <cstring>
  3. #include <string>
  4. #include <algorithm>
  5. #include <map>
  6. using namespace std;
  7. typedef long long ll;
  8. #define inf 1000000
  9. #define mem(a,b) memset(a,b,sizeof(a))
  10. const int N=+;
  11. const int M=*N;
  12. int pre[N],first[N],first2[N],tot,tot2;
  13. bool vis[N];//标记有没有询问
  14. int n;
  15. int fa[N],ans[N],cnt;
  16. int vis2[N],dis[N];
  17. map<string,int>m;
  18.  
  19. struct edge
  20. {
  21. int v,next;
  22. } e[M];
  23. struct Query
  24. {
  25. int v,next,id,flag;//flag用来标记是从左到右还是从到左
  26. } query[M];
  27.  
  28. void add(int u,int v)
  29. {
  30. e[tot].v=v;
  31. e[tot].next =first[u];
  32. first[u]=tot++;
  33. }
  34. void dfs(int u,int len)//建立深度
  35. {
  36. vis2[u]=;
  37. dis[u]=len;
  38. for(int i=first[u];~i;i=e[i].next)
  39. {
  40. int v=e[i].v;
  41. if(!vis2[v])
  42. dfs(v,len+);
  43. }
  44. }
  45.  
  46. void add_query(int u,int v,int id,int flag)//建立询问的边
  47. {
  48. query[tot2].flag =flag;
  49. query[tot2].id=id;
  50. query[tot2].v=v;
  51. query[tot2].next=first2[u];
  52. first2[u]=tot2++;
  53. }
  54.  
  55. int find(int x)
  56. {
  57. return x==pre[x]?x:pre[x]=find(pre[x]);
  58. }
  59.  
  60. void lca(int u,int fa)//Targin算法
  61. {
  62. for(int i=first[u];~i;i=e[i].next)
  63. {
  64. int v=e[i].v;
  65. if(v==fa) continue;
  66. lca(v,u);
  67. pre[v]=u;
  68. }
  69. vis[u]=;
  70. for(int i=first2[u];~i;i=query[i].next)
  71. {
  72. int v=query[i].v;
  73. if(vis[v])//那么find(v)就是最近公共祖先
  74. {
  75. int id=query[i].id;
  76. int flag=query[i].flag ;
  77. if(flag==)
  78. {
  79. if(u==find(v))
  80. {
  81. ans[id]=;//直接跃上找到父目录
  82. }
  83. else if(find(u)==v)
  84. ans[id]=dis[u]-dis[find(v)];//向下走找到目标目录
  85. else
  86. ans[id]=dis[u]-dis[find(v)]+;//一个越到父目录,再向下找
  87. }
  88. else//
  89. {
  90. int u1=v;
  91. int v1=u;
  92. if(u1==find(v1))
  93. ans[id]=;
  94. else if(find(u1)==v1)
  95. ans[id]=dis[u1]-dis[find(v)];
  96. else
  97. ans[id]=dis[u1]-dis[find(v)]+;
  98. }
  99. }
  100. }
  101. }
  102.  
  103. void init()
  104. {
  105. mem(first,-);
  106. mem(first2,-);
  107. mem(vis,);
  108. mem(vis2,);
  109. mem(fa,-);
  110. mem(ans,);
  111. tot=;
  112. tot2=;
  113. cnt=;
  114. m.clear();
  115. for(int i=; i<=n; i++)
  116. pre[i]=i;
  117. }
  118.  
  119. struct node
  120. {
  121. int u,v;
  122. } zz[N];
  123.  
  124. int main()
  125. {
  126. int t,mm;
  127. string s1,s2;
  128. scanf("%d",&t);
  129. while(t--)
  130. {
  131. scanf("%d%d",&n,&mm);
  132. init();
  133. for(int i=; i<n; i++)
  134. {
  135. cin>>s1>>s2;
  136. if(!m[s1]) m[s1]=cnt++;
  137. if(!m[s2]) m[s2]=cnt++;
  138. int a=m[s1],b=m[s2];
  139. add(a,b);
  140. add(b,a);
  141. fa[a]=b;
  142. }
  143. int root=;
  144. while(~fa[root]) root=fa[root];
  145. dfs(root,);
  146. for(int i=; i<=mm; i++)
  147. {
  148. cin>>s1>>s2;
  149. int a=m[s1],b=m[s2];
  150. zz[i].u=a,zz[i].v=b;
  151. add_query(a,b,i,);
  152. add_query(b,a,i,);
  153. }
  154. lca(root,root);
  155. for(int i=; i<=mm; i++)
  156. {
  157. int u=zz[i].u,v=zz[i].v;
  158. if(u==v)
  159. puts("");
  160. else
  161. printf("%d\n",ans[i]);
  162. }
  163. }
  164. return ;
  165. }
 
Source
2013金山西山居创意游戏程序挑战赛——初赛(1)

HDU 4547 CD操作 (LCA最近公共祖先Tarjan模版)的更多相关文章

  1. LCA 最近公共祖先 tarjan离线 总结 结合3个例题

    在网上找了一些对tarjan算法解释较好的文章 并加入了自己的理解 LCA(Least Common Ancestor),顾名思义,是指在一棵树中,距离两个点最近的两者的公共节点.也就是说,在两个点通 ...

  2. LCA 最近公共祖先 Tarjan(离线)算法的基本思路及其算法实现

    首先是最近公共祖先的概念(什么是最近公共祖先?): 在一棵没有环的树上,每个节点肯定有其父亲节点和祖先节点,而最近公共祖先,就是两个节点在这棵树上深度最大的公共的祖先节点. 换句话说,就是两个点在这棵 ...

  3. LCA最近公共祖先——Tarjan模板

    LCA(Lowest Common Ancestors),即最近公共祖先,是指在有根树中,找出某两个结点u和v最近的公共祖先. Tarjan是一种离线算法,时间复杂度O(n+Q),Q表示询问次数,其中 ...

  4. 【HDU 4547 CD操作】LCA问题 Tarjan算法

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=4547 题意:模拟DOS下的cd命令,给出n个节点的目录树以及m次查询,每个查询包含一个当前目录cur和 ...

  5. HDU 4547 CD操作

    传送门 没啥好说的.就是一个LCA. 不过就是有从根到子树里任意一个节点只需要一次操作,特判一下LCA是不是等于v.相等的话不用走.否则就是1次操作. 主要是想写一下倍增的板子. 倍增基于二进制.暴力 ...

  6. LCA最近公共祖先 Tarjan离线算法

    学习博客:  http://noalgo.info/476.html 讲的很清楚! 对于一颗树,dfs遍历时,先向下遍历,并且用并查集维护当前节点和父节点的集合.这样如果关于当前节点(A)的关联节点( ...

  7. LCA(最近公共祖先)专题(不定期更新)

    Tarjan(离线)算法 思路: 1.任选一个点为根节点,从根节点开始. 2.遍历该点u所有子节点v,并标记这些子节点v已被访问过. 3.若是v还有子节点,返回2,否则下一步. 4.合并v到u上. 5 ...

  8. Tarjan算法应用 (割点/桥/缩点/强连通分量/双连通分量/LCA(最近公共祖先)问题)(转载)

    Tarjan算法应用 (割点/桥/缩点/强连通分量/双连通分量/LCA(最近公共祖先)问题)(转载) 转载自:http://hi.baidu.com/lydrainbowcat/blog/item/2 ...

  9. lca 最近公共祖先

    http://poj.org/problem?id=1330 #include<cstdio> #include<cstring> #include<algorithm& ...

随机推荐

  1. XP_SP3_专业汉化版__x86_cd_x14-80404

    1.镜像文件: zh-hans_windows_xp_professional_with_service_pack_3_x86_cd_x14-80404.iso 来自 msdn itellyou 2. ...

  2. istringstream 用法

    istringstream 类用于执行C++风格的串流的输入操作 istringstream用空格作为字符串分隔符 #include <iostream>#include <sstr ...

  3. 智课雅思词汇---二十三、名词性后缀mony

    智课雅思词汇---二十三.名词性后缀mony 一.总结 一句话总结:Latin: action, result of an action or condition; a suffix that for ...

  4. Microsoft Edge Certified with EBS 12.1 and 12.2

    I am very pleased to announce that Microsoft Edge is certified as a new browser for Oracle E-Busines ...

  5. openvswitch的原理和常用命令

    一.Openvswitch工作原理 openvSwitch是一个高质量的.多层虚拟交换机,使用开源Apache2.0许可协议,由 Nicira Networks开发,主要实现代码为可移植的C代码.它的 ...

  6. MySQL 实践

    一.下载MySQL 1.mysql-noinstall-5.1.73-win32.zip 2.mysql-connector-net-6.8.3.msi 二.安装MySQL 三.连接MySQL 1.A ...

  7. LeetCode OJ:Set Matrix Zeroes(设置矩阵0元素)

    Given a m x n matrix, if an element is 0, set its entire row and column to 0. Do it in place. 这题要注意的 ...

  8. String、StringBuffer、StringBuilder分析(jdk8)

    以下代码只挑选了关键方法进行分析 public final class String //String类不可继承,实现了序列化 implements java.io.Serializable, Com ...

  9. Agilent RF fundamentals (11)-Vector modulator

     Vector modulator 矢量调制器:调整信号的幅度和相位 http://www.21ic.com/app/test/201805/762401.htm

  10. 如何在100万文字的文章中 200ms内 快速提取 替换 上万个关键字

    关键点: 关键字 组合 成一棵 hash 树  ( 有属性就直接移动指针到下一个字对象, 没有属性就创建子对象, 再移动指针;  第二次循环在子对象看有没这个属性 ) 探测针       标记结束   ...