4.7 设计并实现一个算法,找出二叉树中某两个结点的第一个共同祖先。不得将额外的结点储存在另外的数据结构中。注意:这不一定是二叉查找树。

解答

本题的关键应当是在Avoid storing additional nodes in a data structure 这句话上。我的理解是,不允许开额外的空间(比如说一个数组)来存储作为中间变量的结点。 虽然我也怀疑它是不是说不允许在结点数据结构Node中加入额外的东西, 比如说父结点的指针。Anyway,我们先从最简单的入手,再一步步加入限制条件。

如果没有任何限制条件,那我觉得最直观的思路就是把其中一个点的所有祖先(包含它自身) 都放入一个哈希表,然后再一步步查找另一个点的祖先结点, 第一个在哈希表中出现的祖先结点即为题目所求。

代码如下,用map模拟(当然,效率比不上哈希表):

算法:

  1. //要使用额外的空间
  2. BinarySearchTree* findFirstAncestor(BinarySearchTree *x,BinarySearchTree *y)
  3. {
  4. if(x==NULL||y==NULL)
  5. return NULL;
  6. map<BinarySearchTree*,bool> mp;
  7. while(x)
  8. {
  9. mp[x]=true;
  10. x=x->parent;
  11. }
  12. while(y)
  13. {
  14. if(mp[y])
  15. return y;
  16. y=y->parent;
  17. }
  18. return y;
  19. }

这里用了一个map来存储中间变量,如果题目不允许开额外的辅助空间,那该如何做呢? 那就老老实实地一个个地试。不断地取出其中一个结点的父结点, 然后判断这个结点是否也为另一个结点的父结点。代码如下:

  1. bool father(BinarySearchTree *x,BinarySearchTree *y)
  2. {
  3. if(x==NULL||y==NULL)
  4. return false;
  5. if(x==y)
  6. return true;
  7. return father(x->left,y)||father(x->right,y);
  8. }
  9. //将每一x的祖先拿出来判断是否为y的祖先,从下到上的方法
  10. BinarySearchTree* find_first_ancestor(BinarySearchTree *x,BinarySearchTree *y)
  11. {
  12. while(x)
  13. {
  14. if(father(x,y))
  15. return x;
  16. x=x->parent;
  17. }
  18. return x;
  19. }

让我们把条件再限制地严苛一些,如果数据结构Node中不允许有指向父亲结点的指针, 那么我们又该如何处理?其实也很简单,首先根结点一定为任意两个结点的共同祖先, 从根结点不断往下找,直到找到最后一个这两结点的共同祖先,即为题目所求。代码如下:

  1. BinarySearchTree* find_ancestor(BinarySearchTree *root,BinarySearchTree *x,BinarySearchTree *y,BinarySearchTree *&ret)
  2. {
  3. if(x==NULL||y==NULL)
  4. return NULL;
  5. if(root&&father(root,x)&&father(root,y))
  6. {
  7. ret=root;
  8. find_ancestor(root->left,x,y,ret);
  9. find_ancestor(root->right,x,y,ret);
  10. }
  11. return ret;
  12. }

这里用到了递归,ans最终保存的是这两个结点从根结点算起最后找到的那个祖先。 因为从根结点开始,每次找到满足要求的结点,ans都会被更新。

C++完整代码:

  1. #include<iostream>
  2. #include<new>
  3. #include<map>
  4. using namespace std;
  5.  
  6. struct BinarySearchTree
  7. {
  8. int elem;
  9. BinarySearchTree *parent;
  10. BinarySearchTree *left;
  11. BinarySearchTree *right;
  12. BinarySearchTree(int x):elem(x),parent(NULL),left(NULL),right(NULL) {}
  13. };
  14.  
  15. void insert(BinarySearchTree *&root,int z)
  16. {
  17. BinarySearchTree *y=new BinarySearchTree(z);
  18. if(root==NULL)
  19. {
  20. root=y;
  21. return;
  22. }
  23. else if(root->left==NULL&&z<root->elem)
  24. {
  25. root->left=y;
  26. y->parent=root;
  27. return;
  28. }
  29. else if(root->right==NULL&&z>root->elem)
  30. {
  31. root->right=y;
  32. y->parent=root;
  33. return;
  34. }
  35. if(z<root->elem)
  36. insert(root->left,z);
  37. else
  38. insert(root->right,z);
  39. }
  40.  
  41. void createBST(BinarySearchTree *&root)
  42. {
  43. int arr[]= {,,,,,,,,,};
  44. for(auto a:arr)
  45. insert(root,a);
  46. }
  47.  
  48. void inorder(BinarySearchTree *root)
  49. {
  50. if(root)
  51. {
  52. inorder(root->left);
  53. cout<<root->elem<<" ";
  54. inorder(root->right);
  55. }
  56. }
  57.  
  58. BinarySearchTree* findMin(BinarySearchTree *root)
  59. {
  60. if(root==NULL||!root->left)
  61. return root;
  62. while(root->left)
  63. {
  64. root=root->left;
  65. }
  66. return root;
  67. }
  68.  
  69. BinarySearchTree* findMax(BinarySearchTree *root)
  70. {
  71. if(root==NULL||!root->right)
  72. return root;
  73. while(root->right)
  74. {
  75. root=root->right;
  76. }
  77. return root;
  78. }
  79.  
  80. BinarySearchTree* findProcessor(BinarySearchTree* x)
  81. {
  82. if(x->left)
  83. return findMax(x->left);
  84. BinarySearchTree *y=x->parent;
  85. while(y&&y->left==x)
  86. {
  87. x=y;
  88. y=x->parent;
  89. }
  90. return y;
  91. }
  92.  
  93. BinarySearchTree* findSuccessor(BinarySearchTree *x)
  94. {
  95. if(x->right)
  96. return findMin(x->right);
  97. BinarySearchTree *y=x->parent;
  98. while(y&&y->right==x)
  99. {
  100. x=y;
  101. y=x->parent;
  102. }
  103. return y;
  104. }
  105. //要使用额外的空间
  106. BinarySearchTree* findFirstAncestor(BinarySearchTree *x,BinarySearchTree *y)
  107. {
  108. if(x==NULL||y==NULL)
  109. return NULL;
  110. map<BinarySearchTree*,bool> mp;
  111. while(x)
  112. {
  113. mp[x]=true;
  114. x=x->parent;
  115. }
  116. while(y)
  117. {
  118. if(mp[y])
  119. return y;
  120. y=y->parent;
  121. }
  122. return y;
  123. }
  124. bool father(BinarySearchTree *x,BinarySearchTree *y)
  125. {
  126. if(x==NULL||y==NULL)
  127. return false;
  128. if(x==y)
  129. return true;
  130. return father(x->left,y)||father(x->right,y);
  131. }
  132. //将每一x的祖先拿出来判断是否为y的祖先,从下到上的方法
  133. BinarySearchTree* find_first_ancestor(BinarySearchTree *x,BinarySearchTree *y)
  134. {
  135. while(x)
  136. {
  137. if(father(x,y))
  138. return x;
  139. x=x->parent;
  140. }
  141. return x;
  142. }
  143. //从上到下的方法
  144. BinarySearchTree* find_ancestor(BinarySearchTree *root,BinarySearchTree *x,BinarySearchTree *y,BinarySearchTree *&ret)
  145. {
  146. if(x==NULL||y==NULL)
  147. return NULL;
  148. if(root&&father(root,x)&&father(root,y))
  149. {
  150. ret=root;
  151. find_ancestor(root->left,x,y,ret);
  152. find_ancestor(root->right,x,y,ret);
  153. }
  154. return ret;
  155. }
  156.  
  157. BinarySearchTree* search(BinarySearchTree* head, int x)
  158. {
  159. if(head == NULL) return NULL;
  160. if(x == head->elem)
  161. return head;
  162. else if(x <= head->elem)
  163. return search(head->left, x);
  164. else
  165. return search(head->right, x);
  166. }
  167. int main()
  168. {
  169. BinarySearchTree *root=NULL;
  170. createBST(root);
  171. inorder(root);
  172. cout<<endl;
  173. BinarySearchTree *n1 = search(root, );
  174. BinarySearchTree*n2 = search(root, );
  175. cout<<n1->elem<<" "<<n2->elem<<endl;
  176. BinarySearchTree *ans = find_first_ancestor(n1, n2);
  177. cout<<ans->elem<<endl;
  178. BinarySearchTree *ans1 = NULL;
  179. find_ancestor(root, n1, n2, ans1);
  180. cout<<ans1->elem<<endl;
  181. BinarySearchTree *pre=findProcessor(n2);
  182. cout<<pre->elem<<endl;
  183. BinarySearchTree *post=findSuccessor(n2);
  184. cout<<post->elem<<endl;
  185. return ;
  186. }

careercup-树与图 4.7的更多相关文章

  1. SqlServer-无限递归树状图结构设计和查询

    在现实生活中,公司的部门设计会涉及到很多子部门,然后子部门下面又存在子部门,形成类似判断的树状结构,比如说评论楼中楼的评论树状图,职位管理的树状图结构等等,实现类似的树状图数据结构是在开发中经常出现的 ...

  2. Android开源图表之树状图和饼状图的官方示例的整理

    最近由于工作需要,所以就在github上搜了下关于chart的三方框架 官方地址https://github.com/PhilJay/MPAndroidChart 由于工作需要我这里整理了一份Ecli ...

  3. D3树状图给指定特性的边特别显示颜色

    D3作为前端图形显示的利器,功能之强,对底层技术细节要求相对比较多. 有一点,就是要理解其基本的数据和节点的匹配规则架构,即enter,update和exit原理,我前面的D3基础篇中有介绍过,不明白 ...

  4. D3树状图异步按需加载数据

    D3.js这个绘图工具,功能强大不必多说,完全一个Data Driven Document的绘图工具,用户可以按照自己的数据以及希望实现的图形,随心所欲的绘图. 图形绘制,D3默认采用的是异步加载,但 ...

  5. [整理] ES5 词法约定文档树状图

    将ES5 词法说明整理为了树状图,方便查阅,请自行点开小图看大图:

  6. bzoj 4871: [Shoi2017]摧毁“树状图” [树形DP]

    4871: [Shoi2017]摧毁"树状图" 题意:一颗无向树,选两条边不重复的路径,删去选择的点和路径剩下一些cc,求最多cc数. update 5.1 : 刚刚发现bzoj上 ...

  7. vue 树状图数据的循环 递归循环

    在main.js中注册一个子组件 在父组件中引用 树状图的数据格式 绑定一个数据传入子组件,子组件props接收数据 子组件中循环调用组件,就实现了递归循环

  8. ztree 文件夹类型的 树状图

    未套程序的源代码: 链接:http://pan.baidu.com/s/1nuHbxhf 密码:4aw2 已套程序的源代码: css样式: /*发布邮件 选择领导弹窗*/ .xuandao{ disp ...

  9. visio画等分树状图

    一 树状图形状 Search里搜索Tree,找到Double Tree或者Multi Tree的形状 二 分出更多branch 按住主干上的黄色小方块,拖出更多分支. 三 等分分支 将每个分支和对应的 ...

  10. ArcGIS教程:树状图

    摘要 构造可显示特征文件里连续合并类之间的属性距离的树示意图(树状图). 使用方法 · 输入特征文件必须採用预定的特征文件格式. 特征文件可使用 Iso 聚类或创建特征工具来创建.该文件必须至少包括两 ...

随机推荐

  1. 145. Binary Tree Postorder Traversal

    题目: Given a binary tree, return the postorder traversal of its nodes' values. For example:Given bina ...

  2. 浅析Android的窗口

    一.窗口的概念 在开发过程中,我们经常会遇到,各种跟窗口相关的类,或者方法.但是,在 Android 的框架设计中,到底什么是窗口?窗口跟 Android Framework 中的 Window 类又 ...

  3. Myeclipse开发内存溢出问题

    MyEclipse开发内存溢出问题   window --> preferences --> MyEclipse --> servers --> Tomcat --> J ...

  4. XML PUBLISHER输出excel禁止自动将数字格式化处理

    方案1: 在 rtf template 里头添加   =T("域"). 方案2: 加个无中断空格  0xA0               In the rtf template, ...

  5. bzoj1497

    这道题让我涨姿势了 对于这类问题,我们称作最大权闭合图问题 就是每个点都有一个点权,要求选择一个点集,其中每个点的指向的点也在点集中,使这样一个点权和最大 对于这种问题,我们添加源点s,汇点t 对于点 ...

  6. MongoDB中ObjectId的误区,以及引起的一系列问题

    近期对两个应用进行改造,在上线过程中出现一系列问题(其中一部分是由于ObjectId误区导致的) 先来了解下ObjectId: TimeStamp 前 4位是一个unix的时间戳,是一个int类别,我 ...

  7. Android 所有颜色代码

    colors.xml <?xml version="1.0" encoding="utf-8" ?> <resources> <c ...

  8. 转载--C++ STL

    转自:http://wenku.baidu.com/view/15d18b4533687e21af45a9a4.html 1.C++ STL 之所以得到广泛的赞誉,也被很多人使用,不只是提供了像vec ...

  9. 【原创】LoadRunner Java Vuser开发环境配置指南

    1 编写目的 本文主要介绍Java运行环境的配置,同时通过编写HelloWorld程序,讲解在LoadRunner下如何开发简单的Java Vuser脚本.关于Java语言的深入学习,大家可以参考其他 ...

  10. 应用程序连接oracle rac

    10g rac vip漂移的目的不是让client不加改变的连接过来而是让client尽快收到一个连接终止的包,然后由client去连接tnsnames中配置的下一个地址.和没有vip相比, 使用vi ...