一共有五种可能的情况:
第五种情况,比如要删除节点7,可以让它的左孩子或者右孩子去继位。这里是让左孩子去继位,左孩子比7小,右孩子比7大,那左孩子应该继位在右孩子的最小的节点的左边,即8左边。然后,让3指向9。
/**
* Definition for a binary tree node.
* public class TreeNode {
* int val;
* TreeNode left;
* TreeNode right;
* TreeNode() {}
* TreeNode(int val) { this.val = val; }
* TreeNode(int val, TreeNode left, TreeNode right) {
* this.val = val;
* this.left = left;
* this.right = right;
* }
* }
*/
class Solution {
public TreeNode deleteNode(TreeNode root, int key) {
if (root == null) return root;
//1.找不到key节点,自动返回原root
if (root.val == key) {
//2.左右都空,说明是叶子,直接删除
if (root.left==null && root.right==null) return null;
//3.左空右不空,右上移
if (root.left==null && root.right!=null) {
root = root.right;
return root;
}
//4.右空左不空,左上移
if (root.left!=null && root.right==null) {
root = root.left;
return root;
}
//5.左右都不空,root的左孩子移到右孩子的最左边
if (root.left!=null && root.right!=null){
TreeNode left = root.left;
root = root.right;
while (root.left!=null){
root = root.left;
}
root.left = left;
return root;
}
}
if (root.val < key) deleteNode(root.left, key);
if (root.val > key) deleteNode(root.right, key);
return root;
}
}
左右都不空时,代码有问题。
代码逻辑不对,没有中间变量cur,相当于少了个变量cur去实现交换操作。
而且,递归处没有赋值给root.left和root.right(因为这里的递归是有返回值TreeNode的)!!!!无法真正实现递归(这一点老是忘记)
/**
* Definition for a binary tree node.
* public class TreeNode {
* int val;
* TreeNode left;
* TreeNode right;
* TreeNode() {}
* TreeNode(int val) { this.val = val; }
* TreeNode(int val, TreeNode left, TreeNode right) {
* this.val = val;
* this.left = left;
* this.right = right;
* }
* }
*/
class Solution {
public TreeNode deleteNode(TreeNode root, int key) {
if (root == null) return root;
//1.找不到key节点,自动返回原root
if (root.val == key) {
//2.左右都空,说明是叶子,直接删除
if (root.left==null && root.right==null) return null;
//3.左空右不空,右上移
if (root.left==null && root.right!=null) {
root = root.right;
return root;
}
//4.右空左不空,左上移
if (root.left!=null && root.right==null) {
root = root.left;
return root;
}
//5.左右都不空,root的左孩子移到右孩子的最左边
if (root.left!=null && root.right!=null){
TreeNode cur = root.right;
while (cur.left!=null){
cur = cur.left;
}
cur.left = root.left;
root = root.right;
return root;
}
}
if (root.val < key) root.left=deleteNode(root.left, key);
if (root.val > key) root.right=deleteNode(root.right, key);
return root;
}
}
递归处逻辑不对。
应该是key比root.val小时,向左搜索
应该是key比root.val大时,向右搜索
/**
* Definition for a binary tree node.
* public class TreeNode {
* int val;
* TreeNode left;
* TreeNode right;
* TreeNode() {}
* TreeNode(int val) { this.val = val; }
* TreeNode(int val, TreeNode left, TreeNode right) {
* this.val = val;
* this.left = left;
* this.right = right;
* }
* }
*/
class Solution {
public TreeNode deleteNode(TreeNode root, int key) {
if (root == null) return root;
//1.找不到key节点,自动返回原root
if (root.val == key) {
//2.左右都空,说明是叶子,直接删除
if (root.left==null && root.right==null) return null;
//3.左空右不空,右上移
if (root.left==null && root.right!=null) {
root = root.right;
return root;
}
//4.右空左不空,左上移
if (root.left!=null && root.right==null) {
root = root.left;
return root;
}
//5.左右都不空,root的左孩子移到右孩子的最左边
if (root.left!=null && root.right!=null){
TreeNode cur = root.right;
while (cur.left!=null){
cur = cur.left;
}
cur.left = root.left;
root = root.right;
return root;
}
}
if (key< root.val) root.left=deleteNode(root.left, key);
if (key> root.val) root.right=deleteNode(root.right, key);
return root;
}
}
O(n),其中 n为 root的节点个数。最差情况下,寻找和删除 cur各需要遍历一次树。
O(n),其中 n为 root的节点个数。递归的深度最深为 O(n)。