抱佛脚-刷题系列之二叉树

抱佛脚一时爽,一直抱佛脚一直爽!这篇文章总结常见的二叉树问题~
参考链接:leetcode 剑指offer cyc的github Annie Kim的博客

一些概念


  • 完全二叉树:堆就是完全二叉树
  • 二叉搜索树:右孩子<=根<左孩子
  • 平衡二叉树: (左子树高度-右子树高度)<=1
  • dfs=先/中/后序遍历
    bfs=层次遍历
    快速排序相当于前序遍历
    归并排序相当于后序遍历

一些经验


  • 一般都有递归、迭代的方法
  • 迭代的方法中:dfs(比如求路径和)用栈,bfs(比如求树高)用队列

各种遍历


层次遍历

  • 非递归
void traverse(TreeNode* root) {
    if (!root) return;
    queue<TreeNode*> myQueue;
    myQueue.push(root);
    while (!myQueue.empty()) {
        TreeNode* cur = myQueue.front();
        visit(cur);
        myQueue.pop();
        if (cur->left) myQueue.push(cur->left);
        if (cur->right) myQueue.push(cur->right);
    }
}
  • 递归
void traverse(TreeNode* root) {
    if (!root) return;
    int depth = getDepth(root);
    for (int i = 1; i <= depth; ++i) visitAtDepth(root, i);
}

int getDepth(TreeNode* root) {
    if (!root) return 0;
    return max(getDepth(root->left), getDepth(root->right)) + 1;
}

void visitAtDepth(TreeNode* root, int depth) {
    if (!root || depth < 1) return;
    if (depth == 1) {
        visit(root);
        return;
    }
    visitAtDepth(root->left, depth - 1);
    visitAtDepth(root->right, depth - 1);
}

中序遍历

  • 非递归
// 写法一
void traverse(TreeNode* root) {
    if (root == nullptr) return;
    stack<TreeNode*> myStack;
    while (!myStack.empty() || root != nullptr) {
        if (root) {
            myStack.push(root);
            root = root->left;
        } else {
            root = myStack.top(); 
            myStack.pop();
            visit(root);
            root = root->right;
        }
    }
}
// 写法二
void traverse(TreeNode* root) {
    if (root == nullptr) return;
    stack<TreeNode*> myStack;
    while (!myStack.empty() || root != nullptr) {
        while (root != nullptr) {
            myStack.push(root);
            root = root->left;
        }
        root = myStack.top();
        visit(root);
        myStack.pop();
        root = root->right;
    }   
}
  • 递归
void traverse(TreeNode* root) {
    if (!root) return;
    traverse(root->left);
    visit(root);
    traverse(root->right);
}
  • morris遍历
void inorderMorrisTraversal(TreeNode *root) {
    TreeNode *cur = root, *prev = NULL;
    while (cur != NULL) {
        if (cur->left == NULL) {
            printf("%d ", cur->val);
            cur = cur->right;
        }
        else {
            // find predecessor
            prev = cur->left;
            while (prev->right != NULL && prev->right != cur)
                prev = prev->right;
            if (prev->right == NULL) {
                prev->right = cur;
                cur = cur->left;
            } else {
                prev->right = NULL;
                printf("%d ", cur->val);
                cur = cur->right;
            }
        }
    }
}

前序遍历

  • 非递归
// 写法一
void traverse(TreeNode* root) {
    if (root == nullptr) return;
    stack<TreeNode*> myStack;
    while (!myStack.empty || root != nullptr) {
        while (root != nullptr) { // 不断把左孩子推入栈
            visit(root);
            myStack.push(root);
            root = root->left;
        }
        root = myStack.top();
        myStack.pop();
        root = root->right; // 前序遍历右子树
    }
}
// 写法二
void traverse(TreeNode* root) {
    if (root == nullptr) return;
    stack<TreeNode*> myStack;
    while (!myStack.empty() || root != nullptr) {
        if (root) {
            myStack.push(root);
            visit(root);
            root = root->left;
        } else {
            root = myStack.top(); 
            myStack.pop();
            root = root->right;
        }
    }
}
  • 递归
void traverse(TreeNode* root) {
    if (!root) return;
    visit(root);
    traverse(root->left);
    traverse(root->right);
}
  • morris遍历
void preorderMorrisTraversal(TreeNode *root) {
    TreeNode *cur = root, *prev = NULL;
    while (cur != NULL) {
        if (cur->left == NULL) {
            printf("%d ", cur->val);
            cur = cur->right;
        } else {
            prev = cur->left;
            while (prev->right != NULL && prev->right != cur)
                prev = prev->right;
            if (prev->right == NULL) {
                printf("%d ", cur->val);  // the only difference with inorder-traversal
                prev->right = cur;
                cur = cur->left;
            } else {
                prev->right = NULL;
                cur = cur->right;
            }
        }
    }
}

后序遍历

  • 非递归
void traverse(TreeNode* root) {
    stack<TreeNode*> myStack;
    TreeNode* lastvisit = root;
    while (!myStack.empty || root != nullptr) {
        if (root != nullptr) {
            myStack.push(root);
            root = root->left;
        } else {
            root = myStack.top();
            if (root->right == nullptr || root->right == lastvisit) {
                visit(root);
                myStack.pop();
                lastvisit = root;
                root = nullptr;
            } else {
                root = root->right;
            }
        }
    }
}
  • 递归
void traverse(TreeNode* root) {
    if (!root) return;
    traverse(root->left);
    traverse(root->right);
    visit(root);
}
  • morris遍历
void reverse(TreeNode *from, TreeNode *to) { // reverse the tree nodes 'from' -> 'to'.
    if (from == to)
        return;
    TreeNode *x = from, *y = from->right, *z;
    while (true) {
        z = y->right;
        y->right = x;
        x = y;
        y = z;
        if (x == to)
            break;
    }
}

void printReverse(TreeNode* from, TreeNode *to) { // print the reversed tree nodes 'from' -> 'to'.
    reverse(from, to);
    TreeNode *p = to;
    while (true) {
        printf("%d ", p->val);
        if (p == from)
            break;
        p = p->right;
    }
    reverse(to, from);
}

void postorderMorrisTraversal(TreeNode *root) {
    TreeNode dump(0);
    dump.left = root;
    TreeNode *cur = &dump, *prev = NULL;
    while (cur) {
        if (cur->left == NULL) {
            cur = cur->right;
        } else {
            prev = cur->left;
            while (prev->right != NULL && prev->right != cur)
                prev = prev->right;
            if (prev->right == NULL) {
                prev->right = cur;
                cur = cur->left;
            } else {
                printReverse(cur->left, prev);  // call print
                prev->right = NULL;
                cur = cur->right;
            }
        }
    }
}

一些题目


求二叉树的直径(lc543)

  • 思路
    1. 包含根的最大直径=左子树的高+右子树的高,不包含根的最大直径=max(左子树的直径,右子树的直径)
    2. 所以,直径=max(左子树的直径,右子树的直径,左子树的高+右子树的高)

前序+中序->二叉树(lc105)

  • 思路
    从前序中找当前树的根节点,中序中根节点的左(右)边就是左(右)子树的节点
class Solution {
public:
    TreeNode *buildTree(vector<int> &preorder, vector<int> &inorder) {
        return buildTree(preorder, 0, preorder.size() - 1, inorder, 0, inorder.size() - 1);
    }

    TreeNode *buildTree(vector<int> &preorder, int pLeft, int pRight, vector<int> &inorder, int iLeft, int iRight) {
        if (pLeft > pRight || iLeft > iRight) return nullptr;
        int i = 0;
        for (i = iLeft; i <= iRight; ++i) {
            if (preorder[pLeft] == inorder[i]) break;
        }
        int leftTreeSize = i - iLeft;
        TreeNode *cur = new TreeNode(preorder[pLeft]);
        cur->left = buildTree(preorder, pLeft + 1, pLeft + leftTreeSize, inorder, iLeft, i - 1);
        cur->right = buildTree(preorder, pLeft + leftTreeSize + 1, pRight, inorder, i + 1, iRight);
        return cur;
    }
};

前序+中序->后序

class Solution {
public:
    void buildTree(vector<int> &preorder, vector<int> &inorder, vector<int>& postorder) {
        buildTree(preorder, 0, preorder.size() - 1, inorder, 0, inorder.size() - 1, postorder);
    }

    void buildTree(vector<int> &preorder, int pLeft, int pRight, vector<int> &inorder, int iLeft, int iRight, vector<int>& postorder) {
        if (pLeft > pRight || iLeft > iRight) return NULL;
        int i = 0;
        for (i = iLeft; i <= iRight; ++i) {
            if (preorder[pLeft] == inorder[i]) break;
        }
        int leftTreeSize = i - iLeft;
        buildTree(preorder, pLeft + 1, pLeft + leftTreeSize, inorder, iLeft, i - 1);
        buildTree(preorder, pLeft + leftTreeSize + 1, pRight, inorder, i + 1, iRight);
        postorder.push_back(inorder[i]);
    }
};

判断二叉树是否对称(lc101)

  • 法一:递归
class Solution {
public:
  bool isSameRoot(TreeNode* root1, TreeNode* root2) {
    if (!root1 && !root2) return true;
    if (!root1 || !root2) return false;
    if (root1->val != root2->val) return false;
    return isSameRoot(root1->left, root2->right) && isSameRoot(root1->right, root2->left);
  }

  bool isSymmetric(TreeNode* root) {
    if (!root) return true;
    return isSameRoot(root->left, root->right);
  }
};
  • 法二:迭代,队列1装左子树层序遍历的结果,队列2装右子树反向层次遍历的结果
class Solution {
public:
  bool isSymmetric(TreeNode* root) {
    if (!root) return true;
    if (!root->left && !root->right) return true;
    if (!root->left || !root->right) return false;
    queue<TreeNode*> q1, q2;
    q1.push(root->left);
    q2.push(root->right);
    while (!q1.empty()) {
      TreeNode* cleft = q1.front(); q1.pop();
      TreeNode* cright = q2.front(); q2.pop();
      if (cleft->val != cright->val) return false;
      if (!cleft->left && cright->right || cleft->left && !cright->right) return false;
      if (!cleft->right && cright->left || cleft->right && !cright->left) return false;
      if (cleft->left) {
        q1.push(cleft->left);
        q2.push(cright->right);
      }
      if (cleft->right) {
        q1.push(cleft->right);
        q2.push(cright->left);
      }
    }
    return true;
  }
};

判断二叉树是否有从根到节点的路径,和为target(lc112)

  • 法一:递归
  bool hasPathSum(TreeNode* root, int sum) {
    if (!root) return false;
    if (!root->left && !root->right) return root->val == sum;
    int flag = false;
    if (root->left) {
      flag = flag || hasPathSum(root->left, sum - root->val);
    }
    if (root->right) {
      flag = flag || hasPathSum(root->right, sum - root->val);
    }
    return flag;
  }
  • 法二:迭代,把根节点的值累加到孩子节点上
  bool hasPathSum(TreeNode* root, int sum) {
    if (!root) return false;
    stack<TreeNode*> s;
    s.push(root);
    while (!s.empty()) {
      TreeNode* cur = s.top();
      s.pop();
      if (!cur->left && !cur->right && cur->val == sum) return true;
      if (cur->left) {
        cur->left->val += cur->val;
        s.push(cur->left);
      }
      if (cur->right) {
        cur->right->val += cur->val;
        s.push(cur->right);
      } 
    }
    return false;
  }

二叉树中和为target的路径(jz24)

class Solution {
public:
    void helper(TreeNode* root, int sum, vector<int>& cur, vector<vector<int>>& result) {
        if (!root) return;
        cur.push_back(root->val);
        if (root->left) helper(root->left, sum - root->val, cur, result);
        if (root->right) helper(root->right, sum - root->val, cur, result);
        if (!root->left && !root->right && sum == root->val) result.push_back(cur);
        cur.pop_back(); 
    }

    vector<vector<int>> pathSum(TreeNode* root, int sum) {
        vector<vector<int>> result;
        vector<int> cur;
        helper(root, sum, cur, result);
        return result;
    }
};

二叉搜索树的最近公共祖先(lc235)

  • 思路:如果根节点的值大于p和q之间的较大值,说明p和q都在左子树中,那么此时我们就进入根节点的左子节点继续递归,如果根节点小于p和q之间的较小值,说明p和q都在右子树中,那么此时我们就进入根节点的右子节点继续递归,如果都不是,则说明当前根节点就是最小共同父节点
TreeNode* lowestCommonAncestor(TreeNode* root, TreeNode* p, TreeNode* q) {
    if (!root) return nullptr;
    while (root) {
        if (root->val > max(p->val, q->val)) root = root->left;
        else if (root->val < min(p->val, q->val)) root = root->right;
        else return root;
    }
    return nullptr;
}

二叉树的最近公共祖先(lc236)

  TreeNode* lowestCommonAncestor(TreeNode* root, TreeNode* p, TreeNode* q) {
    if (!root || p == root || q == root) return root;
    TreeNode* left = lowestCommonAncestor(root->left, p, q);
    TreeNode* right = lowestCommonAncestor(root->right, p, q);
    if (left && right) return root;
    return left ? left : right;
  }

判断是否为二叉搜索树(lc98)

  • 注意:在这道题中定义是左子树值<根<右子树
  • 法一:中序遍历该数到数组中,判断左树是否都小于根、右树是否都大于根、左右树是否都是二叉搜索树
  • 法二:判断左子树的最右节点是否大于根、右子树的最左节点是否小于根、左右树是否都是二叉搜索树
bool isValidBST(TreeNode* root) {
    if (!root) return true;
    TreeNode* rleft = root->left;
    while (rleft && rleft->right) {
        rleft = rleft->right;
    }
    if (rleft && rleft->val >= root->val) return false;
    TreeNode* rright = root->right;
    while (rright && rright->left) {
        rright = rright->left;
    }
    if (rright && rright->val <= root->val) return false;
    return isValidBST(root->left) && isValidBST(root->right);
}

判断一棵树是否另一棵树的子树(jz17)

class Solution {
public:
    bool SubtreeWithRoot(TreeNode* pRoot1, TreeNode* pRoot2) {
        if (!pRoot2) return true;
        if (!pRoot1) return false;
        if (pRoot1->val != pRoot2->val) return false;
        return SubtreeWithRoot(pRoot1->left, pRoot2->left) && SubtreeWithRoot(pRoot1->right, pRoot2->right);
    }
    bool HasSubtree(TreeNode* pRoot1, TreeNode* pRoot2) {
        if (!pRoot2) return false;
        if (!pRoot1) return false;
        return HasSubtree(pRoot1->left, pRoot2) || HasSubtree(pRoot1->right, pRoot2) || SubtreeWithRoot(pRoot1, pRoot2);
    }
};

把二叉树变为其镜像(jz18)

  • 法一:迭代
void Mirror(TreeNode *pRoot) {
  if (!pRoot) return;
  queue<TreeNode*> q;
  q.push(pRoot);
  while (!q.empty()) {
    TreeNode* cur = q.front();
    TreeNode* tmp = cur->left;
    cur->left = cur->right;
    cur->right = tmp;
    if (cur->left) q.push(cur->left);
    if (cur->right) q.push(cur->right);
    q.pop();
  } 
}
  • 法二:递归
void Mirror(TreeNode *pRoot) {
    if (!pRoot) return;
    Mirror(pRoot->left);
    Mirror(pRoot->right);
    TreeNode* tmp = pRoot->left;
    pRoot->left = pRoot->right;
    pRoot->right = tmp;
}

判断数组是否为二叉搜索树后序遍历结果(jz23)

bool VerifySquenceOfBSTHelp(vector<int>& sequence, int left, int right) {
  if (left >= right) return true;
  int root = sequence[right];
  int index = left;
  while (index < right && sequence[index] <= root) ++index;
  for (int i = index; i < right; ++i) {
    if (sequence[i] < root) return false;
  }
  return VerifySquenceOfBSTHelp(sequence, left, index - 1) && VerifySquenceOfBSTHelp(sequence, index, right - 1);
}

bool VerifySquenceOfBST(vector<int> sequence) {
  if (!sequence.size()) return false;
  int n = sequence.size();
  return VerifySquenceOfBSTHelp(sequence, 0, n - 1);
}

二叉树的深度(jz38)

  • 法一:递归,max(左子树深度,右子树深度)+1
  • 法二:迭代,层次遍历算有多少层
int TreeDepth(TreeNode* pRoot) {
    if (!pRoot) return 0;
    queue<TreeNode*> myQueue;
    myQueue.push(pRoot);
    int cnt = 0;
    while (!myQueue.empty()) {
        int width = myQueue.size();
        ++cnt;
        while (width--) {
            TreeNode* cur = myQueue.front();
            myQueue.pop();
            if (cur->left) myQueue.push(cur->left);
            if (cur->right) myQueue.push(cur->right);
        }
    }
    return cnt;
}

二叉树的最小深度(lc111)

  • 法一:递归(注意:考虑到[1,2]的情况,不能直接 return 1 + min(minDepth(root->left), minDepth(root->right)))
int minDepth(TreeNode* root) {
    if (!root) return 0;
    if (!root->left) return 1 + minDepth(root->right);
    if (!root->right) return 1 + minDepth(root->left);
    return 1 + min(minDepth(root->left), minDepth(root->right));
}
  • 法二:迭代
int minDepth(TreeNode* root) {
    if (!root) return 0;
    queue<TreeNode*> q;
    q.push(root);
    int res = 0;
    while (!q.empty()) {
        int width = q.size();
        ++res;
        while (width--) {
            TreeNode* cur = q.front();
            q.pop();
            if (cur->left) q.push(cur->left);
            if (cur->right) q.push(cur->right);
            if (!cur->left && !cur->right) return res;
        }
    }
    return res;
}

判断是否为平衡二叉树(jz39)

  • 思路:递归,abs(左子树深度-右子树深度) <= 1 && 左右子树均是平衡二叉树

二叉树最大路径和(lc124)

  • 思路:对于每个结点来说,要知道经过其左子结点的 path 之和大还是经过右子节点的 path 之和大。递归函数返回值就可以定义为以当前结点为根结点,到叶节点的最大路径之和,然后全局路径最大值放在参数中,用结果 res 来表示
class Solution {
public:
    int maxPathSum(TreeNode* root) {
        int res = INT_MIN;
        helper(root, res);
        return res;
    }
    int helper(TreeNode* node, int& res) { // 返回以node为根节点到叶节点的最大路径和
        if (!node) return 0;
        int left = max(helper(node->left, res), 0);
        int right = max(helper(node->right, res), 0);
        res = max(res, left + right + node->val);
        return max(left, right) + node->val;
    }
};

二叉树展开为链表(lc114)

  • 法一:递归
void flatten(TreeNode* root) {
  if (!root) return;
  if (root->left) flatten(root->left);
  if (root->right) flatten(root->right);
  // 把根节点的左孩子插到根和右孩子之间
  TreeNode* tmp = root->right;
  root->right = root->left;
  root->left = nullptr; // 不要忘了这一句!
  while (root->right) root = root->right;
  root->right = tmp;
}
  • 法二:迭代
void flatten(TreeNode *root) {
    TreeNode *cur = root;
    while (cur) {
        if (cur->left) {
            TreeNode *p = cur->left;
            while (p->right) p = p->right;
            p->right = cur->right;
            cur->right = cur->left;
            cur->left = NULL;
        }
        cur = cur->right;
    }
}

有序数组转平衡二叉树(lc108)

  • 思路:递归,每次取中点作为root,再分别构建左子树和右子树
class Solution {
public:
    TreeNode* sortedArrayToBST(vector<int>& nums, int left, int right) {
      if (right < 0 || right >= nums.size() || left > right) return nullptr;
      int mid = left + (right - left) / 2;
      ListNode* root = new ListNode(nums[mid]);
      ListNode* leftRoot = sortedArrayToBST(nums, left, mid - 1);
      ListNode* rightRoot = sortedArrayToBST(nums, mid + 1, right);
      root->left = leftRoot;
      root->right = rightRoot;
      return root;
    }

    TreeNode* sortedArrayToBST(vector<int>& nums) {
      if (!nums.size()) return nullptr;
      TreeNode* root = sortedArrayToBST(nums, 0, nums.size() - 1);
      return root;
    }
};

返回所有和为target的路径(lc113)

  void findPath(TreeNode* root, int sum, vector<vector<int>>& result, vector<int>& cur) {
    if (!root) return;
    cur.push_back(root->val);
    if (!root->left && !root->right) {
      if (root->val == sum) result.push_back(cur);
    }
    if (root->left) findPath(root->left, sum - root->val, result, cur);
    if (root->right) findPath(root->right, sum - root->val, result, cur);
    cur.pop_back();
  }

  vector<vector<int>> pathSum(TreeNode* root, int sum) {
     vector<vector<int>> result;
     vector<int> cur;
     findPath(root, sum, result, cur);
     return result;
  }

中序遍历的下一个元素

二叉搜索树转为排序的双向链表

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,752评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,100评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,244评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,099评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,210评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,307评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,346评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,133评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,546评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,849评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,019评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,702评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,331评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,030评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,260评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,871评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,898评论 2 351