这篇文章是leetcode
刷题系列的第4
部分——二叉树。这里把有代表性的题目发出来,共计36
道。
leetcode
刷题系列其它文章组织如下:
1
. 数组
2
. 链表
3
. 字符串
4
. 二叉树
5
. 队列和栈
6
. 动态规划
7
. 数据结构设计
8
. 刷题小知识点
二叉树的前序遍历。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 vector <int > preorderTraversal (TreeNode* root) { vector <int > res; stack <TreeNode*> nodes; nodes.push(root); while (!nodes.empty()) { TreeNode* node = nodes.top(); nodes.pop(); while (node) { res.push_back(node->val); if (node->right) { nodes.push(node->right); } node = node->left; } } return res; }
二叉树的中序遍历。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 vector <int > inorderTraversal (TreeNode* root) { stack <TreeNode*> nodes; vector <int > res; while (1 ) { while (root) { nodes.push(root); root = root->left; } if (nodes.empty()) { break ; } root = nodes.top(); nodes.pop(); res.push_back(root->val); root = root->right; } return res; }
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 vector <int > inorderTraversal (TreeNode* root) { stack <TreeNode*> nodes; vector <int > res; while (1 ) { while (root) { nodes.push(root); root = root->right; } if (nodes.empty()) { break ; } root = nodes.top(); nodes.pop(); res.push_back(root->val); root = root->left; } return res; }
二叉树的后序遍历。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 vector <int > postorderTraversal (TreeNode* root) { stack <TreeNode*> nodes; vector <int > res; TreeNode* preVisited = nullptr ; while (1 ) { while (root) { nodes.push(root); root = root->left; } if (nodes.empty()) { break ; } root = nodes.top(); if (root->right == nullptr || root->right == preVisited) { preVisited = root; res.push_back(root->val); nodes.pop(); root = nullptr ; } else { root = root->right; } } return res; }
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 vector <int > postorderTraversal (TreeNode* root) { stack <TreeNode*> nodes; vector <int > res; nodes.push(root); while (!nodes.empty()) { TreeNode* node = nodes.top(); nodes.pop(); while (node) { res.push_back(node->val); if (node->left) { nodes.push(node->left); } node = node->right; } } reverse(res.begin(), res.end()); return res; }
二叉树的层序遍历。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 vector <vector <int >> levelOrder (TreeNode* root) { vector <vector <int >> res; queue <TreeNode*> nodes; if (root) { nodes.push(root); } while (!nodes.empty()) { int n = nodes.size(); vector <int > subVector (n) ; for (int i = 0 ; i < n; i++) { root = nodes.front(); nodes.pop(); subVector[i] = root->val; if (root->left) { nodes.push(root->left); } if (root->right) { nodes.push(root->right); } } res.push_back(subVector); } return res; }
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 vector <vector <int >> levelOrder (TreeNode* root) { vector <vector <int >> res; auto preOrder = [&](auto && preOrder, TreeNode* root, size_t level) { if (root == nullptr ) { return ; } level++; if (res.size() < level) { res.resize(level); } res[level - 1 ].push_back(root->val); preOrder(preOrder, root->left, level); preOrder(preOrder, root->right, level); }; preOrder(preOrder, root, 0 ); return res; }
给定二叉树的根,返回其最大深度。二叉树的最大深度是指沿着从根节点到最远的叶节点的最长路径的节点数。
Example:
1 2 Input: root = [3,9,20,null,null,15,7] Output: 3
1 2 3 4 5 6 7 int maxDepth (TreeNode* root) { if (!root) { return 0 ; } return 1 + max(maxDepth(root->left), maxDepth(root->right)); }
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 int maxDepth (TreeNode* root) { int res = 0 ; function<void (TreeNode*, int )> helper = [&](TreeNode* root, int depth) { if (!root) { return ; } res = max(res, depth); helper(root->left, 1 + depth); helper(root->right, 1 + depth); }; helper(root, 1 ); return res; }
给定二叉树的根,返回其最小深度。二叉树的最小深度是指沿着从根节点到最近的叶节点的最短路径的节点数。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 int minDepth (TreeNode* root) { if (!root) { return 0 ; } int leftMinDepth = minDepth(root->left); int rightMinDepth = minDepth(root->right); if (leftMinDepth == 0 ) { return 1 + rightMinDepth; } if (rightMinDepth == 0 ) { return 1 + leftMinDepth; } return 1 + min(leftMinDepth, rightMinDepth); }
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 int minDepth (TreeNode* root) { queue <TreeNode*> q; if (root) { q.push(root); } int step = 0 ; while (!q.empty()) { int sz = q.size(); while (sz-- > 0 ) { TreeNode* node = q.front(); q.pop(); if (!node->left && !node->right) { return step + 1 ; } if (node->left) { q.push(node->left); } if (node->right) { q.push(node->right); } } step++; } return step; }
给定二叉树的根和一个整数targetSum
,如果树具有从根到叶的路径,则沿路径的所有值加起来等于targetSum
,则返回true
。
Example:
1 2 Input: root = [5,4,8,11,null,13,4,7,2,null,null,null,1], targetSum = 22 Output: true
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 bool hasPathSum (TreeNode* root, int sum) { if (!root) { return false ; } if (!root->left && !root->right) { return sum == root->val; } if (hasPathSum(root->left, sum - root->val)) { return true ; } if (hasPathSum(root->right, sum - root->val)) { return true ; } return false ; }
给定二叉树的根和一个整数targetSum
,返回所有从根到叶的路径,其中每个路径的和等于targetSum
。
Constraints:
The number of nodes in the tree is in the range[0, 5000]
.
-1000 <= Node.val <= 1000
-1000 <= targetSum <= 1000
Example:
1 2 Input: root = [5,4,8,11,null,13,4,7,2,null,null,5,1], targetSum = 22 Output: [[5,4,11,2],[5,8,4,5]]
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 vector <vector <int >> pathSum (TreeNode* root, int sum) { vector <vector <int >> res; vector <int > target; function<void ((TreeNode*, int ))> dfs = [&](TreeNode* root, int sum) { if (!root) { return ; } sum -= root->val; target.push_back(root->val); if (!root->left && !root->right && sum == 0 ) { res.push_back(target); } dfs(root->left, sum); dfs(root->right, sum); target.pop_back(); }; dfs(root, sum); return res; }
你将获得一棵二叉树,其中每个节点都包含一个整数值。查找路径中节点值总和为给定值的路径数。
该路径无需在根或叶处开始或结束,但必须向下(仅从父节点到子节点移动)。
Example:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 root = [10,5,-3,3,2,null,11,3,-2,null,1], sum = 8 10 / \ 5 -3 / \ \ 3 2 11 / \ \ 3 -2 1 Return 3. The paths that sum to 8 are: 1. 5 -> 3 2. 5 -> 2 -> 1 3. -3 -> 11
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 int pathSum (TreeNode* root, int sum) { if (!root) { return 0 ; } return rootPathSum(root, sum) + pathSum(root->left, sum) + pathSum(root->right, sum); } int rootPathSum (TreeNode* root, int targetSum) { int res = 0 ; function<void (TreeNode*, int )> dfs = [&](TreeNode* root, int curSum) { if (root) { curSum += root->val; if (targetSum == curSum) { res++; } dfs(root->left, curSum); dfs(root->right, curSum); } }; dfs(root, 0 ); return res; }
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 int pathSum (TreeNode* root, int sum) { int count = 0 ; unordered_map <int , int > mapping; mapping[0 ] = 1 ; function<void (TreeNode*, int )> dfs = [&](TreeNode* root, int curSum) { if (root) { curSum += root->val; if (mapping.count(curSum - sum)) { count += mapping[curSum - sum]; } mapping[curSum]++; dfs(root->left, curSum); dfs(root->right, curSum); mapping[curSum]--; } }; dfs(root, 0 ); return count; }
对于一棵深度小于5
的树,可以用一组三位十进制整数来表示。
对于每个整数:
百位上的数字表示这个节点的深度D
,1 <= D <= 4
。 十位上的数字表示这个节点在当前层所在的位置P
,1 <= P <= 8
。位置编号与一棵满二叉树的位置编号相同。 个位上的数字表示这个节点的权值V
,0 <= V <= 9
。 给定一个包含三位整数的升序数组,表示一棵深度小于5
的二叉树,请你返回从根到所有叶子结点的路径之和。
示例 1:
1 2 3 4 5 6 7 8 9 输入: [113, 215, 221] 输出: 12 解释: 这棵树形状如下: 3 / \ 5 1 路径和 = (3 + 5) + (3 + 1) = 12.
示例 2:
1 2 3 4 5 6 7 8 9 输入: [113, 221] 输出: 4 解释: 这棵树形状如下: 3 \ 1 路径和 = (3 + 1) = 4.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 int pathSum (vector <int >& nums) { int res = 0 ; unordered_map <int , int > mapping; for (int num : nums) { int depth = num / 100 ; int position = num % 100 / 10 ; int weight = num % 10 ; mapping[pow (2 , depth - 1 ) + position - 1 ] = weight; } function<void (int , int )> dfs = [&](int i, int sum) { if (mapping.count(i) == 0 ) { return ; } sum += mapping[i]; if (mapping.count(2 * i) == 0 && mapping.count(2 * i + 1 ) == 0 ) { res += sum; } dfs(2 * i, sum); dfs(2 * i + 1 , sum); }; dfs(1 , 0 ); return res; }
你将获得只包含0
到9
的数字的二叉树的根。树中的每条从根到叶的路径都代表一个数字。
例如,从根到叶的路径1 -> 2 -> 3
表示数字123
。返回所有从根到叶的数字的总和。
Constraints:
The number of nodes in the tree is in the range[1, 1000]
.
0 <= Node.val <= 9
.
The depth of the tree will not exceed 10
.
Example:
1 2 3 4 5 6 7 Input: root = [4,9,0,5,1] Output: 1026 Explanation: The root-to-leaf path 4->9->5 represents the number 495. The root-to-leaf path 4->9->1 represents the number 491. The root-to-leaf path 4->0 represents the number 40. Therefore, sum = 495 + 491 + 40 = 1026.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 int sumNumbers (TreeNode* root) { int res = 0 ; auto dfs = [&](auto && dfs, auto * root, int num) { if (!root) { return ; } num = num * 10 + root->val; if (!root->left && !root->right) { res += num; } dfs(dfs, root->left, num); dfs(dfs, root->right, num); }; dfs(dfs, root, 0 ); return res; }
二叉树中的路径是节点序列,其中序列中的每对相邻节点都有一条连接它们的边。 一个节点最多只能出现在序列中一次。 请注意,该路径不需要通过根。路径的路径总和是路径中节点值的总和。
给定二叉树的根,返回任何路径的最大路径总和。
Constraints:
The number of nodes in the tree is in the range[1, 3 * 104]
.
-1000 <= Node.val <= 1000
.
Example:
1 2 3 Input: root = [-10,9,20,null,null,15,7] Output: 42 Explanation: The optimal path is 15 -> 20 -> 7 with a path sum of 15 + 20 + 7 = 42.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 int maxPathSum (TreeNode* root) { int res = INT_MIN; auto dfs = [&](auto && dfs, auto * root) { if (!root) { return 0 ; } int leftMaxSum = max(0 , dfs(dfs, root->left)); int rightMaxSum = max(0 , dfs(dfs, root->right)); res = max(res, leftMaxSum + root->val + rightMaxSum); return root->val + max(leftMaxSum, rightMaxSum); }; dfs(dfs, root); return res; }
给定两个整数数组preorder
和inorder
,其中preorder
是二叉树的前序遍历,而inorder
是同一树的中序遍历,构造并返回二叉树。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 TreeNode* buildTree (vector <int >& preorder, vector <int >& inorder) { int n = inorder.size(); auto build = [&](auto && build, int preStart, int inStart, int inEnd) { if (inStart > inEnd) { return (TreeNode*)nullptr ; } int rootVal = preorder[preStart]; TreeNode* root = new TreeNode(rootVal); int index = inStart; for (; index <= inEnd; index++) { if (inorder[index] == rootVal) { break ; } } int leftSize = index - inStart; root->left = build(build, preStart + 1 , inStart, index - 1 ); root->right = build(build, preStart + leftSize + 1 , index + 1 , inEnd); return root; }; return build(build, 0 , 0 , n - 1 ); }
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 TreeNode* buildTree (vector <int >& preorder, vector <int >& inorder) { int n = inorder.size(); unordered_map <int , int > mapping; for (int i = 0 ; i < n; i++) { mapping[inorder[i]] = i; } auto build = [&](auto && build, int rootIndex, int inStart, int inEnd) { if (inStart > inEnd) { return (TreeNode*)nullptr ; } int rootVal = preorder[rootIndex]; int index = mapping[rootVal]; int leftSize = index - inStart; TreeNode* root = new TreeNode(rootVal); root->left = build(build, rootIndex + 1 , inStart, index - 1 ); root->right = build(build, rootIndex + leftSize + 1 , index + 1 , inEnd); return root; }; return build(build, 0 , 0 , n - 1 ); }
给定两个整数数组postorder
和inorder
,其中postorder
是二叉树的后序遍历,而inorder
是同一树的中序遍历,构造并返回二叉树。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 TreeNode* buildTree (vector <int >& inorder, vector <int >& postorder) { int n = inorder.size(); auto build = [&](auto && build, int rootIndex, int inStart, int inEnd) { if (inStart > inEnd) { return (TreeNode*)nullptr ; } int rootVal = postorder[rootIndex]; TreeNode* root = new TreeNode(rootVal); int index = inStart; for (; index <= inEnd; index++) { if (inorder[index] == rootVal) { break ; } } int rightSize = inEnd - index; root->left = build(build, rootIndex - rightSize - 1 , inStart, index - 1 ); root->right = build(build, rootIndex - 1 , index + 1 , inEnd); return root; }; return build(build, n - 1 , 0 , n - 1 ); }
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 TreeNode* buildTree (vector <int >& inorder, vector <int >& postorder) { int n = inorder.size(); unordered_map <int , int > mapping; for (int i = 0 ; i < inorder.size(); i++) { mapping[inorder[i]] = i; } auto build = [&](auto && build, int rootIndex, int inStart, int inEnd) { if (inStart > inEnd) { return (TreeNode*)nullptr ; } int rootVal = postorder[rootIndex]; int index = mapping[rootVal]; int rightSize = inEnd - index; TreeNode* root = new TreeNode(rootVal); root->left = build(build, rootIndex - rightSize - 1 , inStart, index - 1 ); root->right = build(build, rootIndex - 1 , index + 1 , inEnd); return root; }; return build(build, n - 1 , 0 , n - 1 ); }
返回与给定的前序和后序遍历匹配的任意二叉树。二叉树中的值是不同的正整数。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 TreeNode* constructFromPrePost (vector <int >& pre, vector <int >& post) { int n = pre.size(); function<TreeNode*(int , int , int , int )> construct = [&](int preStart, int preEnd, int postStart, int postEnd) { if (preStart > preEnd) { return (TreeNode*)nullptr ; } int rootVal = pre[preStart]; TreeNode* root = new TreeNode(rootVal); if (preStart == preEnd) { return root; } int index = postStart; for (; index < postEnd; index++) { if (post[index] == pre[preStart + 1 ]) { break ; } } int leftSize = index - postStart + 1 ; root->left = construct(preStart + 1 , preStart + leftSize, postStart, index); root->right = construct(preStart + leftSize + 1 , preEnd, index + 1 , postEnd - 1 ); return root; }; return construct(0 , n - 1 , 0 , n - 1 ); }
给定一个整数已排序数组,该数组表示二叉搜索树的前序遍历结果,请构造该BST
树并返回其根。
可以确保始终可以找到满足测试用例要求的BST
。
Constraints:
1 <= preorder.length <= 100
.
1 <= preorder[i] <= 108
.
All the values of preorder
are unique .
Example:
1 2 Input: preorder = [8,5,1,7,10,12] Output: [8,5,10,1,7,null,12]
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 TreeNode* bstFromPreorder (vector <int >& preorder) { int n = preorder.size(); function<TreeNode*(int , int )> build = [&](int preStart, int preEnd) { if (preStart > preEnd) { return (TreeNode*)nullptr ; } int rootVal = preorder[preStart]; TreeNode* root = new TreeNode(rootVal); int index = preStart + 1 ; for (; index <= preEnd; index++) { if (preorder[index] > rootVal) { break ; } } root->left = build(preStart + 1 , index - 1 ); root->right = build(index, preEnd); return root; }; return build(0 , n - 1 ); }
填充二叉树的每个节点下一个指针以指向其下一个右节点。 如果没有下一个右节点,则下一个指针应设置为NULL
。
最初,所有下一个指针都设置为NULL
。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 Node* connect (Node* root) { Node helper; Node* cur = root; while (cur) { Node* last = &helper; while (cur) { if (cur->left) { last->next = cur->left; last = last->next; } if (cur->right) { last->next = cur->right; last = last->next; } cur = cur->next; } cur = helper.next; helper.next = nullptr ; } return root; }
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 Node* connect (Node* root) { Node* cur = root; queue <Node*> nodes; if (cur) { nodes.push(cur); } while (!nodes.empty()) { for (int sz = nodes.size(); sz > 0 ; sz--) { cur = nodes.front(); nodes.pop(); cur->next = sz > 1 ? nodes.front() : nullptr ; if (cur->left) { nodes.push(cur->left); } if (cur->right) { nodes.push(cur->right); } } } return root; }
如果给定的二叉树是棵完美二叉树 :116. Populating Next Right Pointers in Each Node
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 Node* connect (Node* root) { if (!root) { return nullptr ; } function<void (Node*, Node*)> helper = [&](Node* p, Node* q) { if (!p || !q) { return ; } p->next = q; helper(p->left, p->right); helper(q->left, q->right); helper(p->right, q->left); }; helper(root->left, root->right); return root; }
给定二叉树的根,确定它是否是有效的二叉搜索树。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 bool isValidBST (TreeNode* root) { function<bool (TreeNode*, TreeNode*, TreeNode*)> helper = [&](TreeNode* root, TreeNode* min, TreeNode* max) { if (!root) { return true ; } if (min && root->val <= min->val) { return false ; } if (max && root->val >= max->val) { return false ; } return helper(root->left, min, root) && helper(root->right, root, max); }; return helper(root, nullptr , nullptr ); }
给定二叉树的根,找出所有是BST
的子树,分别计算这些子树的所有节点值的和, 返回其中的最大值。
二叉搜索树的定义如下:
任意节点的左子树中的键值都小于此节点的键值。 任意节点的右子树中的键值都大于此节点的键值。 任意节点的左子树和右子树都是二叉搜索树。
Example 1:
1 2 3 Input: root = [1,4,3,2,4,2,5,null,null,null,null,null,null,4,6] Output: 20 Explanation: Maximum sum in a valid Binary search tree is obtained in root node with key equal to 3.
Example 2:
1 2 3 Input: root = [4,3,null,1,2] Output: 2 Explanation: Maximum sum in a valid Binary search tree is obtained in a single root node with key equal to 2.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 int maxSumBST (TreeNode* root) { int res = 0 ; auto postOrder = [&res](auto & postOrder, TreeNode* root) { if (!root) { return make_tuple(0 , INT_MAX, INT_MIN); } auto [leftSum, leftMinVal, leftMaxVal] = postOrder(postOrder, root->left); auto [rightSum, rightMinVal, rightMaxVal] = postOrder(postOrder, root->right); if (leftMaxVal >= root->val || rightMinVal <= root->val) { return make_tuple(0 , INT_MIN, INT_MAX); } int sum = leftSum + root->val + rightSum; res = max(res, sum); return make_tuple(sum, min(root->val, leftMinVal), max(root->val, rightMaxVal)); }; postOrder(postOrder, root); return res; }
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 int maxSumBST (TreeNode* root) { int res = 0 ; auto post = [&](auto && post, auto && root) { if (root == nullptr ) { return make_tuple(true , 0 , INT_MAX, INT_MIN); } auto && [l_isBST, l_sumOfBST, l_minVal, l_maxVal] = post(post, root->left); auto && [r_isBST, r_sumOfBST, r_minVal, r_maxVal] = post(post, root->right); if (!l_isBST || !r_isBST || root->val <= l_maxVal || root->val >= r_minVal) { return make_tuple(false , 0 , 0 , 0 ); } int sum = l_sumOfBST + root->val + r_sumOfBST; res = max(res, sum); return make_tuple(true , sum, min(root->val, l_minVal), max(root->val, r_maxVal)); }; post(post, root); return res; }
序列化是将数据结构或对象转换为位序列的过程,以便可以将其存储在文件或内存缓冲区中,或者通过网络连接链接进行传输,以便稍后在相同或另一台计算机环境中进行重构。
设计一种用于对二叉树进行序列化和反序列化的算法。 序列化和反序列化算法的工作方式没有任何限制。 你只需要确保可以将二叉树序列化为字符串,并且可以将该字符串反序列化为原始树结构。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 class Codec {public : string serialize (TreeNode* root) { string res; serialize(root, res); return res; } TreeNode* deserialize (string data) { stringstream ss (data) ; return deserialize(ss); } private : void serialize (TreeNode* root, string & res) { if (!root) { res += "# " ; return ; } res += to_string(root->val) + " " ; serialize(root->left, res); serialize(root->right, res); } TreeNode* deserialize (stringstream & ss) { if (!ss) { return nullptr ; } string temp; ss >> temp; if (temp == "#" ) { return nullptr ; } TreeNode* root = new TreeNode(stoi(temp)); root->left = deserialize(ss); root->right = deserialize(ss); return root; } };
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 class Codec {public : string serialize (TreeNode* root) { string res; serialize(root, res); return res; } TreeNode* deserialize (string data) { stringstream ss (data) ; vector <string > encoded; string s; while (getline(ss, s, ' ' )) { encoded.push_back(s); } return deserialize(encoded); } private : void serialize (TreeNode* root, string & res) { if (!root) { res += "# " ; return ; } serialize(root->left, res); serialize(root->right, res); res += to_string(root->val) + " " ; } TreeNode* deserialize (vector <string >& encoded) { if (encoded.empty()) { return nullptr ; } string s = encoded.back(); encoded.pop_back(); if (s == "#" ) { return nullptr ; } TreeNode* root = new TreeNode(stoi(s)); root->right = deserialize(encoded); root->left = deserialize(encoded); return root; } };
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 class Codec {public : string serialize (TreeNode* root) { string encoded; queue <TreeNode*> q; q.push(root); while (!q.empty()) { auto node = q.front(); q.pop(); if (node) { encoded += to_string(node->val) + " " ; q.push(node->left); q.push(node->right); } else { encoded += "# " ; } } return encoded; } TreeNode* deserialize (string data) { stringstream ss (data) ; string nodeVal; ss >> nodeVal; if (nodeVal == "#" ) { return nullptr ; } TreeNode* root = new TreeNode(stoi(nodeVal)); queue <TreeNode*> q; q.push(root); while (!q.empty()) { TreeNode* root = q.front(); q.pop(); if (ss >> nodeVal) { if (nodeVal != "#" ) { root->left = new TreeNode(stoi(nodeVal)); q.push(root->left); } } if (ss >> nodeVal) { if (nodeVal != "#" ) { root->right = new TreeNode(stoi(nodeVal)); q.push(root->right); } } } return root; } };
给定一个二叉搜索树和一个目标数,如果BST
中存在两个元素的总和等于给定的目标数,则返回true
。
Example:
1 2 Input: root = [5,3,6,2,4,null,7], k = 9 Output: true
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 bool findTarget (TreeNode* root, int k) { unordered_set <int > setting; auto helper = [&](auto && helper, auto root) { if (!root) { return false ; } if (setting.count(k - root->val)) { return true ; } setting.insert(root->val); return helper(helper, root->left) || helper(helper, root->right); }; return helper(helper, root); }
给定二叉树的根,将树展平为“链表”:“链表”应使用相同的TreeNode
类,其中右子指针指向列表中的下一个节点,而左子指针始终为null
。
“链表”的顺序应与二叉树的前序遍历顺序相同。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 void flatten (TreeNode* root) { if (root) { flatten(root->left); flatten(root->right); TreeNode* temp = root->right; root->right = nullptr ; if (root->left) { root->right = root->left; root->left = nullptr ; } while (root && root->right) { root = root->right; } root->right = temp; } }
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 void flatten (TreeNode* root) { auto cur = root; while (cur) { if (cur->left) { auto pre = cur->left; while (pre->right) { pre = pre->right; } pre->right = cur->right; cur->right = cur->left; cur->left = nullptr ; } cur = cur->right; } }
给定一个原始树和一个克隆树,克隆树是原始树的副本。并给出对原始树中某目标节点的引用。
返回对克隆树中相同节点的引用。
请注意,不允许您更改两个树或目标节点中的任何一个,并且答案必须是对克隆树中节点的引用。
注意:你的解答要适用于在树上允许有重复的值的情况。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 TreeNode* getTargetCopy (TreeNode* original, TreeNode* cloned, TreeNode* target) { if (!original) { return nullptr ; } if (original == target) { return cloned; } TreeNode* found = getTargetCopy(original->left, cloned->left, target); if (!found) { found = getTargetCopy(original->right, cloned->right, target); } return found; }
给定一个二叉搜索树的根节点,恰好树中的两个节点错误地交换了。 在不更改其结构的前提下,恢复这棵二叉搜索树。
Example 1
Example 2
Example 1:
1 2 3 Input: root = [1,3,null,null,2] Output: [3,1,null,null,2] Explanation: 3 cannot be a left child of 1 because 3 > 1. Swapping 1 and 3 makes the BST valid.
Example 2:
1 2 3 Input: root = [3,1,4,null,null,2] Output: [2,1,4,null,null,3] Explanation: 2 cannot be in the right subtree of 3 because 2 < 3. Swapping 2 and 3 makes the BST valid.
注意 :使用O(n)
空间复杂度的解决方案非常简单。 你能设计一个O(1)
空间解决方案吗?
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 void recoverTree (TreeNode* root) { TreeNode* first = nullptr ; TreeNode* second = nullptr ; TreeNode* pre = nullptr ; auto inorder = [&](auto && inorder, auto * cur) { if (cur == nullptr ) { return ; } inorder(inorder, cur->left); if (pre && pre->val > cur->val) { if (first == nullptr ) { first = pre; } second = cur; } pre = cur; inorder(inorder, cur->right); }; inorder(inorder, root); swap(first->val, second->val); }
给定二叉树的根,返回最长路径的长度,其中路径中的每个节点都具有相同的值。 此路径可能会也可能不会通过根。
两个节点之间的路径长度由它们之间的边数表示。
Example 1:
1 2 Input: root = [1,4,5,4,4,5] Output: 2
Example 2:
1 2 Input: root = [5,4,5,1,1,5] Output: 2
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 int longestUnivaluePath (TreeNode* root) { int maxPath = 0 ; auto post = [&](auto && post, auto cur) -> int { if (cur == nullptr ) { return 0 ; } int leftMax = post(post, cur->left); int rightMax = post(post, cur->right); int LeftMaxFromRoot = 0 ; int rightMaxFromRoot = 0 ; if (cur->left && cur->left->val == cur->val) { LeftMaxFromRoot = leftMax + 1 ; } if (cur->right && cur->right->val == cur->val) { rightMaxFromRoot = rightMax + 1 ; } maxPath = max(maxPath, LeftMaxFromRoot + rightMaxFromRoot); return max(LeftMaxFromRoot, rightMaxFromRoot); }; post(post, root); return maxPath; }
给定一个不含重复元素的整数数组。一个以此数组构建的最大二叉树定义如下:
二叉树的根是数组中的最大元素。
左子树是通过数组中最大值左边部分构造出的最大二叉树。
右子树是通过数组中最大值右边部分构造出的最大二叉树。
通过给定的数组构建最大二叉树,并且输出这个树的根节点。
Constraints:
All integers in nums
are unique .
Example:
1 2 Input: nums = [3,2,1,6,0,5] Output: [6,3,5,null,2,0,null,null,1]
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 TreeNode* constructMaximumBinaryTree (vector <int >& nums) { return construct(nums, 0 , nums.size() - 1 ); } TreeNode* construct (vector <int >& nums, int lo, int hi) { if (lo > hi) return nullptr ; int maxIndex = findMax(nums, lo, hi); TreeNode* root = new TreeNode(nums[maxIndex]); root->left = construct(nums, lo, maxIndex - 1 ); root->right = construct(nums, maxIndex + 1 , hi); return root; } int findMax (vector <int >& nums, int lo, int hi) { int maxVal = INT_MIN; int index; for (int i = lo; i <= hi; i++) { if (nums[i] > maxVal) { index = i; maxVal = nums[i]; } } return index; }
最大树定义:一个树,其中每个节点的值都大于其子树中的任何其他值。
给出最大树的根节点root
。
就像之前的654. Maximum Binary Tree
问题那样,给定的树是从列表A
递归地使用下述Construct(A)
例程构造的:
如果A
为空,返回null
否则,令A[i]
作为A
的最大元素。创建一个值为A[i]
的根节点root
root
的左子树将被构建为Construct([A[0], A[1], ..., A[i-1]])
root
的右子树将被构建为Construct([A[i+1], A[i+2], ..., A[A.length - 1]])
返回root
请注意,我们这里没有直接给定A
,只有一个根节点root = Construct(A)
.
假设B
是A
的副本,并在末尾附加值val
。题目数据保证B中的值是不同的。返回Construct(B)
。
Example 1:
1 2 3 Input: root = [4,1,3,null,null,2], val = 5 Output: [5,4,null,1,3,null,null,2] Explanation: A = [1,4,2,3], B = [1,4,2,3,5]
Example 2:
1 2 3 Input: root = [5,2,4,null,1], val = 3 Output: [5,2,4,null,1,null,3] Explanation: A = [2,1,5,4], B = [2,1,5,4,3]
Example 3:
1 2 3 Input: root = [5,2,3,null,1], val = 4 Output: [5,2,4,null,1,3] Explanation: A = [2,1,5,3], B = [2,1,5,3,4]
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 TreeNode* insertIntoMaxTree (TreeNode* root, int val) { if (!root) return new TreeNode(val); if (val > root->val) return new TreeNode(val, root, nullptr ); root->right = insertIntoMaxTree(root->right, val); return root; }
给定一棵树的根,要求你找到最频繁的子树总和。
节点的子树总和定义为由以该节点为根的子树(包括节点本身)形成的所有节点值的总和。 那么,最频繁的子树总和的值是多少?如果有平局,则以任意顺序返回具有最高频率的所有值。
Examples 1 Input:
return [2, -3, 4], since all the values happen only once, return all of them in any order.
Examples 2 Input:
return [2], since 2 happens twice, however -5 only occur once.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 vector <int > findFrequentTreeSum (TreeNode* root) { unordered_map <int , int > mapping; multimap <int , int > mappingSwap; subTreeSum(root, mapping); for (auto elem : mapping) mappingSwap.insert({elem.second, elem.first}); vector <int > res; if (mappingSwap.empty()) return res; int preFreq = mappingSwap.rbegin()->first; for (auto it = mappingSwap.rbegin(); it != mappingSwap.rend(); it++) { if (it->first != preFreq) break ; res.push_back(it->second); } return res; } int subTreeSum (TreeNode* root, unordered_map <int , int >& mapping) { if (!root) return 0 ; int leftSum = subTreeSum(root->left, mapping); int rightSum = subTreeSum(root->right, mapping); int rootSum = leftSum + rightSum + root->val; mapping[rootSum]++; return rootSum; }
给定一棵二叉树,找到树中两个给定节点的最低公共祖先(LCA
)。一个节点的祖先可以是它自己。
Constraints:
所有Node.val
互不相同
p != q
p
和 q
均存在于给定的二叉树中。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 TreeNode* lowestCommonAncestor (TreeNode* root, TreeNode* p, TreeNode* q) { if (!root) return nullptr ; TreeNode* left = lowestCommonAncestor(root->left, p, q); TreeNode* right = lowestCommonAncestor(root->right, p, q); if (root == p || root == q) return root; if (left && right) return root; return left ? left : right; }
给定一棵二叉树的根节点root
,返回给定节点p和q的最近公共祖先(LCA
)节点。如果p
或q
之一不存在于该二叉树中,返回null
。
Constraints:
所有Node.val
互不相同
p != q
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 bool pIsInTree = false ;bool qIsInTree = false ;TreeNode* lowestCommonAncestor (TreeNode* root, TreeNode* p, TreeNode* q) { TreeNode* res = nullptr ; res = helper(root, p, q); if (!pIsInTree || !qIsInTree) return nullptr ; return res; } TreeNode* helper (TreeNode* root, TreeNode* p, TreeNode* q) { if (!root) return nullptr ; TreeNode* left = helper(root->left, p, q); TreeNode* right = helper(root->right, p, q); if (root == p) { pIsInTree = true ; return root; } if (root == q) { qIsInTree = true ; return root; } if (left && right) return root; return left ? left : right; }
给定一棵二叉树中的两个节点p
和q
,返回它们的最近公共祖先节点(LCA
)。
每个节点都包含其父节点的引用(指针)。Node
的定义如下:
1 2 3 4 5 6 7 struct Node { int val; Node* left; Node* right; Node* parent; }
Constraints:
所有Node.val
互不相同
p != q
p
和 q
均存在于给定的二叉树中。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 Node* lowestCommonAncestor (Node* p, Node* q) { unordered_set <Node*> setting; while (p) { setting.insert(p); p = p->parent; } while (q) { if (setting.count(q)) return q; setting.insert(q); q = q->parent; } return nullptr ; }
给定一棵二叉树的根节点root
和TreeNode
类对象的数组(列表)nodes
,返回nodes
中所有节点的最近公共祖先(LCA
)。数组(列表)中所有节点都存在于该二叉树中,且二叉树中所有节点的值都是互不相同的。
Constraints:
所有的Node.val
都是互不相同的。
所有的nodes[i]
都存在于该树中。
所有的nodes[i]
都是互不相同的。
示例 1:
1 2 3 输入: root = [3,5,1,6,2,0,8,null,null,7,4], nodes = [4,7] 输出: 2 解释: 节点 4 和 7 的最近公共祖先是 2。
示例 2:
1 2 3 输入: root = [3,5,1,6,2,0,8,null,null,7,4], nodes = [7,6,2,4] 输出: 5 解释: 节点 7、6、2 和 4 的最近公共祖先节点是 5。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 TreeNode* lowestCommonAncestor (TreeNode* root, vector <TreeNode*>& nodes) { unordered_set <TreeNode*> setting; for (auto node : nodes) setting.insert(node); return dfs(root, setting); } TreeNode* dfs (TreeNode* root, unordered_set <TreeNode*>& setting) { if (!root) return nullptr ; TreeNode* left = dfs(root->left, setting); TreeNode* right = dfs(root->right, setting); if (setting.count(root)) return root; if (left && right) return root; return left ? left : right; }
给定一个二叉搜索树 ,找到该树中两个指定节点的最近公共祖先。
1 2 3 4 5 6 7 8 9 10 11 TreeNode* lowestCommonAncestor (TreeNode* root, TreeNode* p, TreeNode* q) { if (p == root || q == root) return root; if (p->val < root->val && q->val < root->val) return lowestCommonAncestor(root->left, p, q); else if (p->val > root->val && q->val > root->val) return lowestCommonAncestor(root->right, p, q); return root; }
给定二叉树,请检查其是否是其自身的镜像(即围绕其中心对称)。
Example:
1 2 Input: root = [1,2,2,3,4,4,3] Output: true
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 bool isSymmetric (TreeNode* root) { if (!root) return true ; return helper(root->left, root->right); } bool helper (TreeNode* p, TreeNode* q) { if (!p && !q) return true ; if (!p || !q) return false ; if (p->val != q->val) return false ; return helper(p->left, q->right) && helper(p->right, q->left); }
给你一棵以root
为根的二叉树,二叉树中的交错路径定义如下:
选择二叉树中任意节点和一个方向(左或者右)。 如果前进方向为右,那么移动到当前节点的的右子节点,否则移动到它的左子节点。 改变前进方向:左变右或者右变左。 重复第二步和第三步,直到你在树中无法继续移动。 交错路径的长度定义为:访问过的节点数目 - 1
(单个节点的路径长度为0
)。
请你返回给定树中最长交错路径的长度。
Example 1
Example 2
Example 1:
1 2 3 Input: root = [1,null,1,1,1,null,null,1,1,null,1,null,null,null,1,null,1] Output: 3 Explanation: Longest ZigZag path in blue nodes (right -> left -> right).
Example 2:
1 2 3 Input: root = [1,1,1,null,1,null,null,1,1,null,1] Output: 4 Explanation: Longest ZigZag path in blue nodes (left -> right -> left -> right).
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 int res = 0 ;int longestZigZag (TreeNode* root) { helper(root->left, 0 , 1 ); helper(root->right, 1 , 1 ); return res; } void helper (TreeNode* node, bool dir, int depth) { if (!node) return ; res = max(res, depth); if (dir) { helper(node->left, 0 , depth + 1 ); helper(node->right, 1 , 1 ); } else { helper(node->right, 1 , depth + 1 ); helper(node->left, 0 , 1 ); } } int res = 0 ;int helper (TreeNode* root, bool dir) { if (!root) return 0 ; int tmp_L = 1 + helper(root->left, 0 ); int tmp_R = 1 + helper(root->right, 1 ); res = max(res, tmp_L); res = max(res, tmp_R); return dir ? tmp_L : tmp_R; } int longestZigZag (TreeNode* root) { helper(root, 0 ); return res - 1 ; }
实现一个二叉搜索树迭代器类BSTIterator
,表示一个按中序遍历二叉搜索树(BST
)的迭代器:
BSTIterator(TreeNode* root)
初始化BSTIterator
类的一个对象。BST
的根节点root
会作为构造函数的一部分给出。指针应初始化为一个不存在于BST
中的数字,且该数字小于BST
中的任何元素。
bool hasNext()
如果向指针右侧遍历存在数字,则返回true
;否则返回false
。 int next()
将指针向右移动,然后返回指针处的数字。注意,指针初始化为一个不存在于BST
中的数字,所以对next()
的首次调用将返回BST
中的最小元素。
你可以假设next()
调用总是有效的,也就是说,当调用next()
时,BST
的中序遍历中至少存在一个下一个数字。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 class BSTIterator { public : BSTIterator(TreeNode* root) : _pointer(root) {} int next () { while (_pointer) { _stack.push(_pointer); _pointer = _pointer->left; } _pointer = _stack.top(); int val = _pointer->val; _stack.pop(); _pointer = _pointer->right; return val; } bool hasNext () { return _pointer || !_stack.empty(); } private : TreeNode* _pointer; stack <TreeNode*> _stack; };