目录
一、(leetcode 513)找树左下角的值
二、(leetcode 112) 路径总和
三、(leetcode 113)路径总和ii
四、(leetcode 106)从中序与后序遍历序列构造二叉树
五、(leetcode 106)从前序与中序遍历序列构造二叉树
力扣题目链接
状态:想到了要用到回溯法,但是困在了之前的套路,不知从何下手。
该题也用到了回溯法,但是值得注意的是回溯法不一定需要进行路径的记录。更进一步说的话,回溯法本身和路径记录没有关系,因为利用递归进行遍历本身就包含了回溯的过程,区别在于普通的遍历考虑的是当前节点和子节点的关系,而回溯则要求还要考虑和其父节点的关系。所以说,回溯法中,具体是怎么控制前进,抑或是怎么得到答案,全在于如何发现可以终止和进行回溯的条件是什么。
class Solution {
public:
int max_depth = INT_MIN;
int res;
void traversal(TreeNode* root, int depth){
if(root->left == nullptr && root->right == nullptr){
if(depth > max_depth){
max_depth = depth;
res = root->val;
}
return;
}
if(root->left){
traversal(root->left, depth+1);
}
if(root->right){
traversal(root->right, depth+1);
}
}
int findBottomLeftValue(TreeNode* root) {
int depth = 0;
traversal(root, depth);
return res;
}
};
力扣题目链接
状态:Debug后AC。
和上一题类似,注意节点回溯的吃入和吐出的逻辑
class Solution {
public:
int sum_num = 0, target = 0;
bool res = false;
void traversal(TreeNode* node){
if(res || node == nullptr) {return;}
if(node->left == nullptr && node->right == nullptr){
if(sum_num+node->val == target){
res = true;
}else{
return;
}
}
if(node->left){
sum_num += node->val;
traversal(node->left);
sum_num -= node->val;
}
if(node->right){
sum_num += node->val;
traversal(node->right);
sum_num -= node->val;
}
return;
}
bool hasPathSum(TreeNode* root, int targetSum) {
target = targetSum;
traversal(root);
return res;
}
};
力扣题目链接
状态:Debug后AC。
和路径总和在思路上一样,但是多了一些判断操作
class Solution {
public:
int target = 0;
int sum(vector& path){
int res = 0;
for(auto num : path){
res += num;
}
//cout << res << endl;
return res;
}
void traversal(TreeNode* node, vector& path, vector>& res){
if(node->left == nullptr && node->right == nullptr){
//cout << node->val << ": ";
path.emplace_back(node->val);
int temp = sum(path);
if(temp == target){
res.emplace_back(path);
}
path.pop_back();
return;
}
if(node->left){
path.emplace_back(node->val);
traversal(node->left, path, res);
path.pop_back();
}
if(node->right){
path.emplace_back(node->val);
traversal(node->right, path, res);
path.pop_back();
}
return;
}
vector> pathSum(TreeNode* root, int targetSum) {
vector> res;
if(root == nullptr) return res;
vector path;
//path.emplace_back(root->val);
target = targetSum;
traversal(root, path, res);
return res;
}
};
力扣题目链接
状态:思路容易想到,但是具体的实现需要一步一步拆解。
通过后序遍历数组的最后一位来确定当前的根节点,接着确定根节点在中序遍历数组中的位置(索引),根据索引可以将中序遍历数组划分为左右两个子数组。通过中序遍历的子数组可以同样地确定后序遍历中左右子数组的划分。这其中要注意的是左右子数组的边界问题。
class Solution {
private:
TreeNode* traversal (vector& inorder, vector& postorder) {
if (postorder.size() == 0) return NULL;
// 后序遍历数组最后一个元素,就是当前的中间节点
int rootValue = postorder[postorder.size() - 1];
TreeNode* root = new TreeNode(rootValue);
// 叶子节点
if (postorder.size() == 1) return root;
// 找到中序遍历的切割点
int delimiterIndex;
for (delimiterIndex = 0; delimiterIndex < inorder.size(); delimiterIndex++) {
if (inorder[delimiterIndex] == rootValue) break;
}
// 切割中序数组
// 左闭右开区间:[0, delimiterIndex)
vector leftInorder(inorder.begin(), inorder.begin() + delimiterIndex);
// [delimiterIndex + 1, end)
vector rightInorder(inorder.begin() + delimiterIndex + 1, inorder.end() );
// postorder 舍弃末尾元素
postorder.resize(postorder.size() - 1);
// 切割后序数组
// 依然左闭右开,注意这里使用了左中序数组大小作为切割点
// [0, leftInorder.size)
vector leftPostorder(postorder.begin(), postorder.begin() + leftInorder.size());
// [leftInorder.size(), end)
vector rightPostorder(postorder.begin() + leftInorder.size(), postorder.end());
root->left = traversal(leftInorder, leftPostorder);
root->right = traversal(rightInorder, rightPostorder);
return root;
}
public:
TreeNode* buildTree(vector& inorder, vector& postorder) {
if (inorder.size() == 0 || postorder.size() == 0) return NULL;
return traversal(inorder, postorder);
}
};
力扣题目链接
状态:上一题方法中有太多vector的创建,可以通过改成索引的方法来优化。
class Solution {
private:
TreeNode* traversal (vector& inorder, int inorderBegin, int inorderEnd, vector& preorder, int preorderBegin, int preorderEnd) {
if (preorderBegin == preorderEnd) return NULL;
int rootValue = preorder[preorderBegin]; // 注意用preorderBegin 不要用0
TreeNode* root = new TreeNode(rootValue);
if (preorderEnd - preorderBegin == 1) return root;
int delimiterIndex;
for (delimiterIndex = inorderBegin; delimiterIndex < inorderEnd; delimiterIndex++) {
if (inorder[delimiterIndex] == rootValue) break;
}
// 切割中序数组
// 中序左区间,左闭右开[leftInorderBegin, leftInorderEnd)
int leftInorderBegin = inorderBegin;
int leftInorderEnd = delimiterIndex;
// 中序右区间,左闭右开[rightInorderBegin, rightInorderEnd)
int rightInorderBegin = delimiterIndex + 1;
int rightInorderEnd = inorderEnd;
// 切割前序数组
// 前序左区间,左闭右开[leftPreorderBegin, leftPreorderEnd)
int leftPreorderBegin = preorderBegin + 1;
int leftPreorderEnd = preorderBegin + 1 + delimiterIndex - inorderBegin; // 终止位置是起始位置加上中序左区间的大小size
// 前序右区间, 左闭右开[rightPreorderBegin, rightPreorderEnd)
int rightPreorderBegin = preorderBegin + 1 + (delimiterIndex - inorderBegin);
int rightPreorderEnd = preorderEnd;
root->left = traversal(inorder, leftInorderBegin, leftInorderEnd, preorder, leftPreorderBegin, leftPreorderEnd);
root->right = traversal(inorder, rightInorderBegin, rightInorderEnd, preorder, rightPreorderBegin, rightPreorderEnd);
return root;
}
public:
TreeNode* buildTree(vector& preorder, vector& inorder) {
if (inorder.size() == 0 || preorder.size() == 0) return NULL;
// 参数坚持左闭右开的原则
return traversal(inorder, 0, inorder.size(), preorder, 0, preorder.size());
}
};