二叉树作为一种重要的数据结构,在算法领域有着广泛的应用,而深度优先搜索(DFS)是二叉树遍历和操作的核心算法之一。通过 DFS,可以以递归或迭代的方式深入探索树的每一个节点,并高效地解决路径查找、节点计数、最大深度等问题。在这篇文章中,我们将深入剖析二叉树的深搜算法,从基础概念到典型应用,再到代码实现,带你全面掌握这一重要的算法工具。
题目链接:https://leetcode.cn/problems/evaluate-boolean-binary-tree/
bool evaluateTree(TreeNode* root)
evaluateTree(TreeNode* root->left)
和evaluateTreel(TreeNode* root->right)
,无条件相信它一定能帮我们得到左右孩子的布尔值我们结合示例分析:
/**
* Definition for a binary tree node.
* struct TreeNode {
* int val;
* TreeNode *left;
* TreeNode *right;
* TreeNode() : val(0), left(nullptr), right(nullptr) {}
* TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
* TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
* };
*/
class Solution {
public:
bool evaluateTree(TreeNode* root) {
if(!root->left) return root->val == 0? false: true;
bool left = evaluateTree(root->left);
bool right = evaluateTree(root->right);
return root->val == 2? left | right: left & right;
}
};
题目链接:https://leetcode.cn/problems/sum-root-to-leaf-numbers/
如图所示:
/**
* Definition for a binary tree node.
* struct TreeNode {
* int val;
* TreeNode *left;
* TreeNode *right;
* TreeNode() : val(0), left(nullptr), right(nullptr) {}
* TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
* TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
* };
*/
class Solution {
public:
int dfs(TreeNode* root, int prenum){
// 1.接收父节点的值并加上自己
int nownum = prenum * 10 + root->val;
// 2.检测叶子结点,设置递归终止条件
if(root->left == nullptr && root->right == nullptr) return nownum;
// 3.传值给左右孩子
int ret = 0;
if(root->left) ret += dfs(root->left, nownum);
if(root->right) ret += dfs(root->right, nownum);
// 返回相加后的值
return ret;
}
int sumNumbers(TreeNode* root) {
return dfs(root, 0);
}
};
题目链接:https://leetcode.cn/problems/binary-tree-pruning/
通过决策树分析:
Node* dfs(root)
/**
* Definition for a binary tree node.
* struct TreeNode {
* int val;
* TreeNode *left;
* TreeNode *right;
* TreeNode() : val(0), left(nullptr), right(nullptr) {}
* TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
* TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
* };
*/
class Solution {
public:
TreeNode* pruneTree(TreeNode* root) {
if(root == nullptr) return nullptr;
root->left = pruneTree(root->left);
root->right = pruneTree(root->right);
if(root->left == nullptr && root->right == nullptr && root->val == 0)
root = nullptr;
return root;
}
};
题目链接:https://leetcode.cn/problems/validate-binary-search-tree/
如图分析:
/**
* Definition for a binary tree node.
* struct TreeNode {
* int val;
* TreeNode *left;
* TreeNode *right;
* TreeNode() : val(0), left(nullptr), right(nullptr) {}
* TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
* TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
* };
*/
class Solution {
long prev = LONG_MIN; // 定义并初始化全局变量prev,用于记录遍历过程中访问过的节点的最大值。
public:
bool isValidBST(TreeNode* root) {
if(!root) return true; // 如果当前节点为空(即已经遍历到叶子节点的下一个位置),则返回true。因为在二叉搜索树的定义中,空树被认为是有效的二叉搜索树。
bool left = isValidBST(root->left);
// 剪枝
if(!left) return false;
bool cur = false;
if(prev < root->val)
cur = true;
prev = root->val;
// 剪枝
if(!cur) return false;
bool right = isValidBST(root->right);
return left && right && cur; // 如果左子树、右子树以及当前节点都满足二叉搜索树的条件,则返回true,表示整个树是有效的二叉搜索树。
}
};
题目链接:https://leetcode.cn/problems/kth-smallest-element-in-a-bst/description/
count
记录需要跳过的节点数,当 count == 0
时,当前节点即为目标值。count
用于记录剩余需要遍历的节点数,初始值为 k
。ret
存储最终找到的第 k 小元素。dfs
: root == NULL
),直接返回。dfs(root->left)
。count--
。count == 0
),如果是,将当前节点值存入 ret
。dfs(root->right)
。kthSmallest
方法中,设置 count = k
,然后调用 dfs(root)
开始中序遍历。ret
中存储的就是第 k 小的元素。ret
,即目标值。如图分析:
/**
* Definition for a binary tree node.
* struct TreeNode {
* int val;
* TreeNode *left;
* TreeNode *right;
* TreeNode() : val(0), left(nullptr), right(nullptr) {}
* TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
* TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
* };
*/
class Solution {
public:
int count;
int ret = 0;
void dfs(TreeNode* root) {
if (!root || count == 0) return;
dfs(root->left);
count--;
if (count == 0) ret = root->val;
dfs(root->right);
}
int kthSmallest(TreeNode* root, int k) {
count = k;
dfs(root);
return ret;
}
};
。
(k)。
。
;最坏情况下
(退化为链表)。
题目链接:https://leetcode.cn/problems/binary-tree-paths/
vector<string> ret
存储所有路径的结果。string path
存储当前路径的字符串。ret
中。"->"
表示继续延续路径。ret
。如图分析:
/**
* Definition for a binary tree node.
* struct TreeNode {
* int val;
* TreeNode *left;
* TreeNode *right;
* TreeNode() : val(0), left(nullptr), right(nullptr) {}
* TreeNode(int x) : val(x), left(nullptr), right(nullptr) {}
* TreeNode(int x, TreeNode *left, TreeNode *right) : val(x), left(left), right(right) {}
* };
*/
class Solution {
public:
vector<string> ret;
vector<string> binaryTreePaths(TreeNode* root) {
string path;
dfs(root, path);
return ret;
}
void dfs(TreeNode* root, string path){
if(!root) return;
if(!root->left && !root->right){
path += to_string(root->val);
ret.push_back(path);
return;
}
else{
path += to_string(root->val) + "-" + ">";
}
dfs(root->left, path);
dfs(root->right, path);
}
};
,其中
是节点总数。
,其中
是路径长度。由于二叉树的深度最大为
,总复杂度为
。
,因此空间复杂度为
。
ret
占用的空间为 。
深度优先搜索不仅是二叉树操作的基础算法,更是一种处理递归结构问题的通用策略。通过对 DFS 的深入理解和实践,可以在许多复杂问题中找到高效的解决方案。从基础到应用,我们希望这篇文章帮助你更好地掌握 DFS 算法,并在未来的编程之路上将其灵活运用到各类数据结构和问题中。记住,算法的艺术在于实践,而实践则在于深度的探索!
今天的分享到这里就结束啦!如果觉得文章还不错的话,可以三连支持一下,17的主页还有很多有趣的文章,欢迎小伙伴们前去点评,您的支持就是17前进的动力!