当前位置:   article > 正文

【数据结构基础】树 - 二叉搜索树(BST)_bst数据结构

bst数据结构
本文主要介绍 二叉树中最基本的二叉查找树(Binary Search Tree),(又:二叉搜索树,二叉排序树)它或者是一棵空树,或者是具有下列性质的二叉树: 若它的左子树不空,则左子树上所有结点的值均小于它的根结点的值; 若它的右子树不空,则右子树上所有结点的值均大于它的根结点的值; 它的左、右子树也分别为二叉排序树。

BST的定义

二叉查找树中:

  • 若任意节点的左子树不空,则左子树上所有结点的值均小于它的根结点的值;

  • 任意节点的右子树不空,则右子树上所有结点的值均大于它的根结点的值;

  • 任意节点的左、右子树也分别为二叉查找树。

  • 没有键值相等的节点。

动画效果请参考 BST

BST的实现

节点

BSTree是二叉树,它保存了二叉树的根节点mRoot;mRoot是BSTNode类型,而BSTNode是二叉查找树的节点,它是BSTree的内部类。BSTNode包含二叉查找树的几个基本信息:

  • key -- 它是关键字,是用来对二叉查找树的节点进行排序的。

  • left -- 它指向当前节点的左孩子。

  • right -- 它指向当前节点的右孩子。

  • parent -- 它指向当前节点的父结点。

  1. public class BSTree<T extends Comparable<T>> {
  2. private BSTNode<T> mRoot; // 根结点
  3. public class BSTNode<T extends Comparable<T>> {
  4. T key; // 关键字(键值)
  5. BSTNode<T> left; // 左孩子
  6. BSTNode<T> right; // 右孩子
  7. BSTNode<T> parent; // 父结点
  8. public BSTNode(T key, BSTNode<T> parent, BSTNode<T> left, BSTNode<T> right) {
  9. this.key = key;
  10. this.parent = parent;
  11. this.left = left;
  12. this.right = right;
  13. }
  14. }
  15. ......
  16. }

遍历

这里讲解前序遍历、中序遍历、后序遍历3种方式。

前序遍历

若二叉树非空,则执行以下操作:

  • 访问根结点;

  • 先序遍历左子树;

  • 先序遍历右子树。

  1. private void preOrder(BSTNode<T> tree) {
  2. if(tree != null) {
  3. System.out.print(tree.key+" ");
  4. preOrder(tree.left);
  5. preOrder(tree.right);
  6. }
  7. }
  8. public void preOrder() {
  9. preOrder(mRoot);
  10. }
中序遍历

若二叉树非空,则执行以下操作:

  • 中序遍历左子树;

  • 访问根结点;

  • 中序遍历右子树。

  1. private void inOrder(BSTNode<T> tree) {
  2. if(tree != null) {
  3. inOrder(tree.left);
  4. System.out.print(tree.key+" ");
  5. inOrder(tree.right);
  6. }
  7. }
  8. public void inOrder() {
  9. inOrder(mRoot);
  10. }
后序遍历

若二叉树非空,则执行以下操作:

  • 后序遍历左子树;

  • 后序遍历右子树;

  • 访问根结点。

  1. private void postOrder(BSTNode<T> tree) {
  2. if(tree != null)
  3. {
  4. postOrder(tree.left);
  5. postOrder(tree.right);
  6. System.out.print(tree.key+" ");
  7. }
  8. }
  9. public void postOrder() {
  10. postOrder(mRoot);
  11. }

看看下面这颗树的各种遍历方式:

对于上面的二叉树而言,

  • 前序遍历结果: 8 3 1 6 4 7 10 14 13

  • 中序遍历结果: 1 3 4 6 7 8 10 13 14

  • 后序遍历结果: 1 4 7 6 3 13 14 10 8

查找

  • 递归版本的代码

  1. /*
  2. * (递归实现)查找"二叉树x"中键值为key的节点
  3. */
  4. private BSTNode<T> search(BSTNode<T> x, T key) {
  5. if (x==null)
  6. return x;
  7. int cmp = key.compareTo(x.key);
  8. if (cmp < 0)
  9. return search(x.left, key);
  10. else if (cmp > 0)
  11. return search(x.right, key);
  12. else
  13. return x;
  14. }
  15. public BSTNode<T> search(T key) {
  16. return search(mRoot, key);
  17. }
  • 非递归版本的代码

  1. /*
  2. * (非递归实现)查找"二叉树x"中键值为key的节点
  3. */
  4. private BSTNode<T> iterativeSearch(BSTNode<T> x, T key) {
  5. while (x!=null) {
  6. int cmp = key.compareTo(x.key);
  7. if (cmp < 0)
  8. x = x.left;
  9. else if (cmp > 0)
  10. x = x.right;
  11. else
  12. return x;
  13. }
  14. return x;
  15. }
  16. public BSTNode<T> iterativeSearch(T key) {
  17. return iterativeSearch(mRoot, key);
  18. }

最大值和最小值

  • 查找最大结点

  1. /*
  2. * 查找最大结点: 返回tree为根结点的二叉树的最大结点。
  3. */
  4. private BSTNode<T> maximum(BSTNode<T> tree) {
  5. if (tree == null)
  6. return null;
  7. while(tree.right != null)
  8. tree = tree.right;
  9. return tree;
  10. }
  11. public T maximum() {
  12. BSTNode<T> p = maximum(mRoot);
  13. if (p != null)
  14. return p.key;
  15. return null;
  16. }
  • 查找最小结点

  1. /*
  2. * 查找最小结点: 返回tree为根结点的二叉树的最小结点。
  3. */
  4. private BSTNode<T> minimum(BSTNode<T> tree) {
  5. if (tree == null)
  6. return null;
  7. while(tree.left != null)
  8. tree = tree.left;
  9. return tree;
  10. }
  11. public T minimum() {
  12. BSTNode<T> p = minimum(mRoot);
  13. if (p != null)
  14. return p.key;
  15. return null;
  16. }

前驱和后继

节点的前驱: 是该节点的左子树中的最大节点。 节点的后继: 是该节点的右子树中的最小节点。

  • 查找前驱节点

  1. /*
  2. * 找结点(x)的前驱结点。即,查找"二叉树中数据值小于该结点"的"最大结点"。
  3. */
  4. public BSTNode<T> predecessor(BSTNode<T> x) {
  5. // 如果x存在左孩子,则"x的前驱结点"为 "以其左孩子为根的子树的最大结点"。
  6. if (x.left != null)
  7. return maximum(x.left);
  8. // 如果x没有左孩子。则x有以下两种可能:
  9. // (01) x是"一个右孩子",则"x的前驱结点"为 "它的父结点"。
  10. // (01) x是"一个左孩子",则查找"x的最低的父结点,并且该父结点要具有右孩子",找到的这个"最低的父结点"就是"x的前驱结点"。
  11. BSTNode<T> y = x.parent;
  12. while ((y!=null) && (x==y.left)) {
  13. x = y;
  14. y = y.parent;
  15. }
  16. return y;
  17. }
  • 查找后继节点

  1. /*
  2. * 找结点(x)的后继结点。即,查找"二叉树中数据值大于该结点"的"最小结点"。
  3. */
  4. public BSTNode<T> successor(BSTNode<T> x) {
  5. // 如果x存在右孩子,则"x的后继结点"为 "以其右孩子为根的子树的最小结点"。
  6. if (x.right != null)
  7. return minimum(x.right);
  8. // 如果x没有右孩子。则x有以下两种可能:
  9. // (01) x是"一个左孩子",则"x的后继结点"为 "它的父结点"。
  10. // (02) x是"一个右孩子",则查找"x的最低的父结点,并且该父结点要具有左孩子",找到的这个"最低的父结点"就是"x的后继结点"。
  11. BSTNode<T> y = x.parent;
  12. while ((y!=null) && (x==y.right)) {
  13. x = y;
  14. y = y.parent;
  15. }
  16. return y;
  17. }

插入

  1. /*
  2. * 将结点插入到二叉树中
  3. *
  4. * 参数说明:
  5. * tree 二叉树的
  6. * z 插入的结点
  7. */
  8. private void insert(BSTree<T> bst, BSTNode<T> z) {
  9. int cmp;
  10. BSTNode<T> y = null;
  11. BSTNode<T> x = bst.mRoot;
  12. // 查找z的插入位置
  13. while (x != null) {
  14. y = x;
  15. cmp = z.key.compareTo(x.key);
  16. if (cmp < 0)
  17. x = x.left;
  18. else
  19. x = x.right;
  20. }
  21. z.parent = y;
  22. if (y==null)
  23. bst.mRoot = z;
  24. else {
  25. cmp = z.key.compareTo(y.key);
  26. if (cmp < 0)
  27. y.left = z;
  28. else
  29. y.right = z;
  30. }
  31. }
  32. /*
  33. * 新建结点(key),并将其插入到二叉树中
  34. *
  35. * 参数说明:
  36. * tree 二叉树的根结点
  37. * key 插入结点的键值
  38. */
  39. public void insert(T key) {
  40. BSTNode<T> z=new BSTNode<T>(key,null,null,null);
  41. // 如果新建结点失败,则返回。
  42. if (z != null)
  43. insert(this, z);
  44. }

删除

  1. /*
  2. * 删除结点(z),并返回被删除的结点
  3. *
  4. * 参数说明:
  5. * bst 二叉树
  6. * z 删除的结点
  7. */
  8. private BSTNode<T> remove(BSTree<T> bst, BSTNode<T> z) {
  9. BSTNode<T> x=null;
  10. BSTNode<T> y=null;
  11. if ((z.left == null) || (z.right == null) )
  12. y = z;
  13. else
  14. y = successor(z);
  15. if (y.left != null)
  16. x = y.left;
  17. else
  18. x = y.right;
  19. if (x != null)
  20. x.parent = y.parent;
  21. if (y.parent == null)
  22. bst.mRoot = x;
  23. else if (y == y.parent.left)
  24. y.parent.left = x;
  25. else
  26. y.parent.right = x;
  27. if (y != z)
  28. z.key = y.key;
  29. return y;
  30. }
  31. /*
  32. * 删除结点(z),并返回被删除的结点
  33. *
  34. * 参数说明:
  35. * tree 二叉树的根结点
  36. * z 删除的结点
  37. */
  38. public void remove(T key) {
  39. BSTNode<T> z, node;
  40. if ((z = search(mRoot, key)) != null)
  41. if ( (node = remove(this, z)) != null)
  42. node = null;
  43. }

打印

  1. /*
  2. * 打印"二叉查找树"
  3. *
  4. * key -- 节点的键值
  5. * direction -- 0,表示该节点是根节点;
  6. * -1,表示该节点是它的父结点的左孩子;
  7. * 1,表示该节点是它的父结点的右孩子。
  8. */
  9. private void print(BSTNode<T> tree, T key, int direction) {
  10. if(tree != null) {
  11. if(direction==0) // tree是根节点
  12. System.out.printf("%2d is root\n", tree.key);
  13. else // tree是分支节点
  14. System.out.printf("%2d is %2d's %6s child\n", tree.key, key, direction==1?"right" : "left");
  15. print(tree.left, tree.key, -1);
  16. print(tree.right,tree.key, 1);
  17. }
  18. }
  19. public void print() {
  20. if (mRoot != null)
  21. print(mRoot, mRoot.key, 0);
  22. }

销毁

  1. /*
  2. * 销毁二叉树
  3. */
  4. private void destroy(BSTNode<T> tree) {
  5. if (tree==null)
  6. return ;
  7. if (tree.left != null)
  8. destroy(tree.left);
  9. if (tree.right != null)
  10. destroy(tree.right);
  11. tree=null;
  12. }
  13. public void clear() {
  14. destroy(mRoot);
  15. mRoot = null;
  16. }

测试程序

下面对测试程序的流程进行分析!

  • 新建"二叉查找树"root。

  • 向二叉查找树中依次插入1,5,4,3,2,6 。如下图所示:

  • 遍历和查找

插入1,5,4,3,2,6之后,得到的二叉查找树如下:

  1. 前序遍历结果: 154326
  2. 中序遍历结果: 123456
  3. 后序遍历结果: 234651
  4. 最小值是1,而最大值是6。
  • 删除节点4。如下图所示:

  • 重新遍历该二叉查找树。

中序遍历结果: 1 2 4 5 6

代码和测试代码

代码实现

  1. /**
  2. * Java 语言: 二叉查找树
  3. *
  4. * @author skywang
  5. * @date 2013/11/07
  6. */
  7. public class BSTree<T extends Comparable<T>> {
  8. private BSTNode<T> mRoot; // 根结点
  9. public class BSTNode<T extends Comparable<T>> {
  10. T key; // 关键字(键值)
  11. BSTNode<T> left; // 左孩子
  12. BSTNode<T> right; // 右孩子
  13. BSTNode<T> parent; // 父结点
  14. public BSTNode(T key, BSTNode<T> parent, BSTNode<T> left, BSTNode<T> right) {
  15. this.key = key;
  16. this.parent = parent;
  17. this.left = left;
  18. this.right = right;
  19. }
  20. public T getKey() {
  21. return key;
  22. }
  23. public String toString() {
  24. return "key:"+key;
  25. }
  26. }
  27. public BSTree() {
  28. mRoot=null;
  29. }
  30. /*
  31. * 前序遍历"二叉树"
  32. */
  33. private void preOrder(BSTNode<T> tree) {
  34. if(tree != null) {
  35. System.out.print(tree.key+" ");
  36. preOrder(tree.left);
  37. preOrder(tree.right);
  38. }
  39. }
  40. public void preOrder() {
  41. preOrder(mRoot);
  42. }
  43. /*
  44. * 中序遍历"二叉树"
  45. */
  46. private void inOrder(BSTNode<T> tree) {
  47. if(tree != null) {
  48. inOrder(tree.left);
  49. System.out.print(tree.key+" ");
  50. inOrder(tree.right);
  51. }
  52. }
  53. public void inOrder() {
  54. inOrder(mRoot);
  55. }
  56. /*
  57. * 后序遍历"二叉树"
  58. */
  59. private void postOrder(BSTNode<T> tree) {
  60. if(tree != null)
  61. {
  62. postOrder(tree.left);
  63. postOrder(tree.right);
  64. System.out.print(tree.key+" ");
  65. }
  66. }
  67. public void postOrder() {
  68. postOrder(mRoot);
  69. }
  70. /*
  71. * (递归实现)查找"二叉树x"中键值为key的节点
  72. */
  73. private BSTNode<T> search(BSTNode<T> x, T key) {
  74. if (x==null)
  75. return x;
  76. int cmp = key.compareTo(x.key);
  77. if (cmp < 0)
  78. return search(x.left, key);
  79. else if (cmp > 0)
  80. return search(x.right, key);
  81. else
  82. return x;
  83. }
  84. public BSTNode<T> search(T key) {
  85. return search(mRoot, key);
  86. }
  87. /*
  88. * (非递归实现)查找"二叉树x"中键值为key的节点
  89. */
  90. private BSTNode<T> iterativeSearch(BSTNode<T> x, T key) {
  91. while (x!=null) {
  92. int cmp = key.compareTo(x.key);
  93. if (cmp < 0)
  94. x = x.left;
  95. else if (cmp > 0)
  96. x = x.right;
  97. else
  98. return x;
  99. }
  100. return x;
  101. }
  102. public BSTNode<T> iterativeSearch(T key) {
  103. return iterativeSearch(mRoot, key);
  104. }
  105. /*
  106. * 查找最小结点: 返回tree为根结点的二叉树的最小结点。
  107. */
  108. private BSTNode<T> minimum(BSTNode<T> tree) {
  109. if (tree == null)
  110. return null;
  111. while(tree.left != null)
  112. tree = tree.left;
  113. return tree;
  114. }
  115. public T minimum() {
  116. BSTNode<T> p = minimum(mRoot);
  117. if (p != null)
  118. return p.key;
  119. return null;
  120. }
  121. /*
  122. * 查找最大结点: 返回tree为根结点的二叉树的最大结点。
  123. */
  124. private BSTNode<T> maximum(BSTNode<T> tree) {
  125. if (tree == null)
  126. return null;
  127. while(tree.right != null)
  128. tree = tree.right;
  129. return tree;
  130. }
  131. public T maximum() {
  132. BSTNode<T> p = maximum(mRoot);
  133. if (p != null)
  134. return p.key;
  135. return null;
  136. }
  137. /*
  138. * 找结点(x)的后继结点。即,查找"二叉树中数据值大于该结点"的"最小结点"。
  139. */
  140. public BSTNode<T> successor(BSTNode<T> x) {
  141. // 如果x存在右孩子,则"x的后继结点"为 "以其右孩子为根的子树的最小结点"。
  142. if (x.right != null)
  143. return minimum(x.right);
  144. // 如果x没有右孩子。则x有以下两种可能:
  145. // (01) x是"一个左孩子",则"x的后继结点"为 "它的父结点"。
  146. // (02) x是"一个右孩子",则查找"x的最低的父结点,并且该父结点要具有左孩子",找到的这个"最低的父结点"就是"x的后继结点"。
  147. BSTNode<T> y = x.parent;
  148. while ((y!=null) && (x==y.right)) {
  149. x = y;
  150. y = y.parent;
  151. }
  152. return y;
  153. }
  154. /*
  155. * 找结点(x)的前驱结点。即,查找"二叉树中数据值小于该结点"的"最大结点"。
  156. */
  157. public BSTNode<T> predecessor(BSTNode<T> x) {
  158. // 如果x存在左孩子,则"x的前驱结点"为 "以其左孩子为根的子树的最大结点"。
  159. if (x.left != null)
  160. return maximum(x.left);
  161. // 如果x没有左孩子。则x有以下两种可能:
  162. // (01) x是"一个右孩子",则"x的前驱结点"为 "它的父结点"。
  163. // (01) x是"一个左孩子",则查找"x的最低的父结点,并且该父结点要具有右孩子",找到的这个"最低的父结点"就是"x的前驱结点"。
  164. BSTNode<T> y = x.parent;
  165. while ((y!=null) && (x==y.left)) {
  166. x = y;
  167. y = y.parent;
  168. }
  169. return y;
  170. }
  171. /*
  172. * 将结点插入到二叉树中
  173. *
  174. * 参数说明:
  175. * tree 二叉树的
  176. * z 插入的结点
  177. */
  178. private void insert(BSTree<T> bst, BSTNode<T> z) {
  179. int cmp;
  180. BSTNode<T> y = null;
  181. BSTNode<T> x = bst.mRoot;
  182. // 查找z的插入位置
  183. while (x != null) {
  184. y = x;
  185. cmp = z.key.compareTo(x.key);
  186. if (cmp < 0)
  187. x = x.left;
  188. else
  189. x = x.right;
  190. }
  191. z.parent = y;
  192. if (y==null)
  193. bst.mRoot = z;
  194. else {
  195. cmp = z.key.compareTo(y.key);
  196. if (cmp < 0)
  197. y.left = z;
  198. else
  199. y.right = z;
  200. }
  201. }
  202. /*
  203. * 新建结点(key),并将其插入到二叉树中
  204. *
  205. * 参数说明:
  206. * tree 二叉树的根结点
  207. * key 插入结点的键值
  208. */
  209. public void insert(T key) {
  210. BSTNode<T> z=new BSTNode<T>(key,null,null,null);
  211. // 如果新建结点失败,则返回。
  212. if (z != null)
  213. insert(this, z);
  214. }
  215. /*
  216. * 删除结点(z),并返回被删除的结点
  217. *
  218. * 参数说明:
  219. * bst 二叉树
  220. * z 删除的结点
  221. */
  222. private BSTNode<T> remove(BSTree<T> bst, BSTNode<T> z) {
  223. BSTNode<T> x=null;
  224. BSTNode<T> y=null;
  225. if ((z.left == null) || (z.right == null) )
  226. y = z;
  227. else
  228. y = successor(z);
  229. if (y.left != null)
  230. x = y.left;
  231. else
  232. x = y.right;
  233. if (x != null)
  234. x.parent = y.parent;
  235. if (y.parent == null)
  236. bst.mRoot = x;
  237. else if (y == y.parent.left)
  238. y.parent.left = x;
  239. else
  240. y.parent.right = x;
  241. if (y != z)
  242. z.key = y.key;
  243. return y;
  244. }
  245. /*
  246. * 删除结点(z),并返回被删除的结点
  247. *
  248. * 参数说明:
  249. * tree 二叉树的根结点
  250. * z 删除的结点
  251. */
  252. public void remove(T key) {
  253. BSTNode<T> z, node;
  254. if ((z = search(mRoot, key)) != null)
  255. if ( (node = remove(this, z)) != null)
  256. node = null;
  257. }
  258. /*
  259. * 销毁二叉树
  260. */
  261. private void destroy(BSTNode<T> tree) {
  262. if (tree==null)
  263. return ;
  264. if (tree.left != null)
  265. destroy(tree.left);
  266. if (tree.right != null)
  267. destroy(tree.right);
  268. tree=null;
  269. }
  270. public void clear() {
  271. destroy(mRoot);
  272. mRoot = null;
  273. }
  274. /*
  275. * 打印"二叉查找树"
  276. *
  277. * key -- 节点的键值
  278. * direction -- 0,表示该节点是根节点;
  279. * -1,表示该节点是它的父结点的左孩子;
  280. * 1,表示该节点是它的父结点的右孩子。
  281. */
  282. private void print(BSTNode<T> tree, T key, int direction) {
  283. if(tree != null) {
  284. if(direction==0) // tree是根节点
  285. System.out.printf("%2d is root\n", tree.key);
  286. else // tree是分支节点
  287. System.out.printf("%2d is %2d's %6s child\n", tree.key, key, direction==1?"right" : "left");
  288. print(tree.left, tree.key, -1);
  289. print(tree.right,tree.key, 1);
  290. }
  291. }
  292. public void print() {
  293. if (mRoot != null)
  294. print(mRoot, mRoot.key, 0);
  295. }
  296. }

测试代码

  1. /**
  2. * Java 语言: 二叉查找树
  3. *
  4. * @author skywang
  5. * @date 2013/11/07
  6. */
  7. public class BSTreeTest {
  8. private static final int arr[] = {1,5,4,3,2,6};
  9. public static void main(String[] args) {
  10. int i, ilen;
  11. BSTree<Integer> tree=new BSTree<Integer>();
  12. System.out.print("== 依次添加: ");
  13. ilen = arr.length;
  14. for(i=0; i<ilen; i++) {
  15. System.out.print(arr[i]+" ");
  16. tree.insert(arr[i]);
  17. }
  18. System.out.print("\n== 前序遍历: ");
  19. tree.preOrder();
  20. System.out.print("\n== 中序遍历: ");
  21. tree.inOrder();
  22. System.out.print("\n== 后序遍历: ");
  23. tree.postOrder();
  24. System.out.println();
  25. System.out.println("== 最小值: "+ tree.minimum());
  26. System.out.println("== 最大值: "+ tree.maximum());
  27. System.out.println("== 树的详细信息: ");
  28. tree.print();
  29. System.out.print("\n== 删除根节点: "+ arr[3]);
  30. tree.remove(arr[3]);
  31. System.out.print("\n== 中序遍历: ");
  32. tree.inOrder();
  33. System.out.println();
  34. // 销毁二叉树
  35. tree.clear();
  36. }
  37. }

测试结果

  1. == 依次添加: 1 5 4 3 2 6
  2. == 前序遍历: 1 5 4 3 2 6
  3. == 中序遍历: 1 2 3 4 5 6
  4. == 后序遍历: 2 3 4 6 5 1
  5. == 最小值: 1
  6. == 最大值: 6
  7. == 树的详细信息:
  8. is root
  9. is 1's right child
  10. is 5's left child
  11. is 4's left child
  12. is 3's left child
  13. is 5's right child
  14. == 删除根节点: 3
  15. == 中序遍历: 1 2 4 5 6

BST相关题目

二叉查找树(BST): 根节点大于等于左子树所有节点,小于等于右子树所有节点。

二叉查找树中序遍历有序。

修剪二叉查找树

669. Trim a Binary Search Tree (Easy)

  1. Input:
  2. 3
  3. / \
  4. 0 4
  5. \
  6. 2
  7. /
  8. 1
  9. L = 1
  10. R = 3
  11. Output:
  12. 3
  13. /
  14. 2
  15. /
  16. 1

题目描述: 只保留值在 L ~ R 之间的节点

  1. public TreeNode trimBST(TreeNode root, int L, int R) {
  2. if (root == null) return null;
  3. if (root.val > R) return trimBST(root.left, L, R);
  4. if (root.val < L) return trimBST(root.right, L, R);
  5. root.left = trimBST(root.left, L, R);
  6. root.right = trimBST(root.right, L, R);
  7. return root;
  8. }

寻找二叉查找树的第 k 个元素

230. Kth Smallest Element in a BST (Medium)

中序遍历解法:

  1. private int cnt = 0;
  2. private int val;
  3. public int kthSmallest(TreeNode root, int k) {
  4. inOrder(root, k);
  5. return val;
  6. }
  7. private void inOrder(TreeNode node, int k) {
  8. if (node == null) return;
  9. inOrder(node.left, k);
  10. cnt++;
  11. if (cnt == k) {
  12. val = node.val;
  13. return;
  14. }
  15. inOrder(node.right, k);
  16. }

递归解法:

  1. public int kthSmallest(TreeNode root, int k) {
  2. int leftCnt = count(root.left);
  3. if (leftCnt == k - 1) return root.val;
  4. if (leftCnt > k - 1) return kthSmallest(root.left, k);
  5. return kthSmallest(root.right, k - leftCnt - 1);
  6. }
  7. private int count(TreeNode node) {
  8. if (node == null) return 0;
  9. return 1 + count(node.left) + count(node.right);
  10. }

把二叉查找树每个节点的值都加上比它大的节点的值

Convert BST to Greater Tree (Easy)

  1. Input: The root of a Binary Search Tree like this:
  2. 5
  3. / \
  4. 2 13
  5. Output: The root of a Greater Tree like this:
  6. 18
  7. / \
  8. 20 13

先遍历右子树。

  1. private int sum = 0;
  2. public TreeNode convertBST(TreeNode root) {
  3. traver(root);
  4. return root;
  5. }
  6. private void traver(TreeNode node) {
  7. if (node == null) return;
  8. traver(node.right);
  9. sum += node.val;
  10. node.val = sum;
  11. traver(node.left);
  12. }

二叉查找树的最近公共祖先

235. Lowest Common Ancestor of a Binary Search Tree (Easy)

  1. _______6______
  2. / \
  3. ___2__ ___8__
  4. / \ / \
  5. 0 4 7 9
  6. / \
  7. 3 5
  8. For example, the lowest common ancestor (LCA) of nodes 2 and 8 is 6. Another example is LCA of nodes 2 and 4 is 2, since a node can be a descendant of itself according to the LCA definition.
  1. public TreeNode lowestCommonAncestor(TreeNode root, TreeNode p, TreeNode q) {
  2. if (root.val > p.val && root.val > q.val) return lowestCommonAncestor(root.left, p, q);
  3. if (root.val < p.val && root.val < q.val) return lowestCommonAncestor(root.right, p, q);
  4. return root;
  5. }

二叉树的最近公共祖先

236. Lowest Common Ancestor of a Binary Tree (Medium)

  1. _______3______
  2. / \
  3. ___5__ ___1__
  4. / \ / \
  5. 6 2 0 8
  6. / \
  7. 7 4
  8. For example, the lowest common ancestor (LCA) of nodes 5 and 1 is 3. Another example is LCA of nodes 5 and 4 is 5, since a node can be a descendant of itself according to the LCA definition.
  1. public TreeNode lowestCommonAncestor(TreeNode root, TreeNode p, TreeNode q) {
  2. if (root == null || root == p || root == q) return root;
  3. TreeNode left = lowestCommonAncestor(root.left, p, q);
  4. TreeNode right = lowestCommonAncestor(root.right, p, q);
  5. return left == null ? right : right == null ? left : root;
  6. }

从有序数组中构造二叉查找树

108. Convert Sorted Array to Binary Search Tree (Easy)

  1. public TreeNode sortedArrayToBST(int[] nums) {
  2. return toBST(nums, 0, nums.length - 1);
  3. }
  4. private TreeNode toBST(int[] nums, int sIdx, int eIdx){
  5. if (sIdx > eIdx) return null;
  6. int mIdx = (sIdx + eIdx) / 2;
  7. TreeNode root = new TreeNode(nums[mIdx]);
  8. root.left = toBST(nums, sIdx, mIdx - 1);
  9. root.right = toBST(nums, mIdx + 1, eIdx);
  10. return root;
  11. }

根据有序链表构造平衡的二叉查找树

109. Convert Sorted List to Binary Search Tree (Medium)

  1. Given the sorted linked list: [-10,-3,0,5,9],
  2. One possible answer is: [0,-3,9,-10,null,5], which represents the following height balanced BST:
  3. 0
  4. / \
  5. -3 9
  6. / /
  7. -10 5
  1. public TreeNode sortedListToBST(ListNode head) {
  2. if (head == null) return null;
  3. if (head.next == null) return new TreeNode(head.val);
  4. ListNode preMid = preMid(head);
  5. ListNode mid = preMid.next;
  6. preMid.next = null; // 断开链表
  7. TreeNode t = new TreeNode(mid.val);
  8. t.left = sortedListToBST(head);
  9. t.right = sortedListToBST(mid.next);
  10. return t;
  11. }
  12. private ListNode preMid(ListNode head) {
  13. ListNode slow = head, fast = head.next;
  14. ListNode pre = head;
  15. while (fast != null && fast.next != null) {
  16. pre = slow;
  17. slow = slow.next;
  18. fast = fast.next.next;
  19. }
  20. return pre;
  21. }

在二叉查找树中寻找两个节点,使它们的和为一个给定值

653. Two Sum IV - Input is a BST (Easy)

  1. Input:
  2. 5
  3. / \
  4. 3 6
  5. / \ \
  6. 2 4 7
  7. Target = 9
  8. Output: True

使用中序遍历得到有序数组之后,再利用双指针对数组进行查找。

应该注意到,这一题不能用分别在左右子树两部分来处理这种思想,因为两个待求的节点可能分别在左右子树中。

  1. public boolean findTarget(TreeNode root, int k) {
  2. List<Integer> nums = new ArrayList<>();
  3. inOrder(root, nums);
  4. int i = 0, j = nums.size() - 1;
  5. while (i < j) {
  6. int sum = nums.get(i) + nums.get(j);
  7. if (sum == k) return true;
  8. if (sum < k) i++;
  9. else j--;
  10. }
  11. return false;
  12. }
  13. private void inOrder(TreeNode root, List<Integer> nums) {
  14. if (root == null) return;
  15. inOrder(root.left, nums);
  16. nums.add(root.val);
  17. inOrder(root.right, nums);
  18. }

在二叉查找树中查找两个节点之差的最小绝对值

530. Minimum Absolute Difference in BST (Easy)

  1. Input:
  2. 1
  3. \
  4. 3
  5. /
  6. 2
  7. Output:
  8. 1

利用二叉查找树的中序遍历为有序的性质,计算中序遍历中临近的两个节点之差的绝对值,取最小值。

  1. private int minDiff = Integer.MAX_VALUE;
  2. private TreeNode preNode = null;
  3. public int getMinimumDifference(TreeNode root) {
  4. inOrder(root);
  5. return minDiff;
  6. }
  7. private void inOrder(TreeNode node) {
  8. if (node == null) return;
  9. inOrder(node.left);
  10. if (preNode != null) minDiff = Math.min(minDiff, node.val - preNode.val);
  11. preNode = node;
  12. inOrder(node.right);
  13. }

寻找二叉查找树中出现次数最多的值

501. Find Mode in Binary Search Tree (Easy)

  1. 1
  2. \
  3. 2
  4. /
  5. 2
  6. return [2].

答案可能不止一个,也就是有多个值出现的次数一样多。

  1. private int curCnt = 1;
  2. private int maxCnt = 1;
  3. private TreeNode preNode = null;
  4. public int[] findMode(TreeNode root) {
  5. List<Integer> maxCntNums = new ArrayList<>();
  6. inOrder(root, maxCntNums);
  7. int[] ret = new int[maxCntNums.size()];
  8. int idx = 0;
  9. for (int num : maxCntNums) {
  10. ret[idx++] = num;
  11. }
  12. return ret;
  13. }
  14. private void inOrder(TreeNode node, List<Integer> nums) {
  15. if (node == null) return;
  16. inOrder(node.left, nums);
  17. if (preNode != null) {
  18. if (preNode.val == node.val) curCnt++;
  19. else curCnt = 1;
  20. }
  21. if (curCnt > maxCnt) {
  22. maxCnt = curCnt;
  23. nums.clear();
  24. nums.add(node.val);
  25. } else if (curCnt == maxCnt) {
  26. nums.add(node.val);
  27. }
  28. preNode = node;
  29. inOrder(node.right, nums);
  30. }
声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/Cpp五条/article/detail/274925
推荐阅读
相关标签
  

闽ICP备14008679号