Day 23 669. 修剪二叉搜索树 108.将有序数组转换为二叉搜索树 538.把二叉搜索树转换为累加树 总结篇

修剪二叉搜索树

给定一个二叉搜索树,同时给定最小边界L 和最大边界 R。通过修剪二叉搜索树,使得所有节点的值在[L, R]中 (R>=L) 。你可能需要改变树的根节点,所以结果应当返回修剪好的二叉搜索树的新的根节点。

669.修剪二叉搜索树

669.修剪二叉搜索树1

​ 最直接的想法,遍历树然后找到root->val在[L,R]以外的节点删除,通过递归处理返回根节点;

​ 如果此时简单的给出这种代码:

       root->left = trimBST(root->left, low, high);
       root->right = trimBST(root->right, low, high);

​ 就会忽视掉当前节点不在范围内但是节点左(右)子树可能在范围内的情况;

​ 如下图所示:

​ 也就是在删除节点时,还要对其子树进行判断;至于实现方式可以参考上一题的子树嫁接方法,无须重构二叉树结构;

​ 下面写递归三部曲:

​ 首先确定递归函数参数和返回值:遍历整棵树,做修改,其实不需要返回值也可以完成修剪(其实就是从二叉树中移除节点)的操作;但是有返回值,更方便,可以通过递归函数的返回值来移除节点,不需要额外操作;

​ 其次确定终止条件:修剪的操作并不是在终止条件进行的,所以就是遇到空节点返回就可以了;

​ 最后确定单层递归的逻辑:嫁接子树,此处不再赘述;

​ 递归代码如下:

	TreeNode* traversal(TreeNode* root, int L, int R){
        if(root == NULL)	return NULL;
        if(root->val < L){//节点值小于左边边界值
            TreeNode* tempNode = traversal(root->right, L, R);//此节点右子树中寻找所有大于L的值的节点,继续嵌套递归修剪右子树;
            return tempNode;
        }
        if(root->val > R)	return	traversal(root->left, L, R);//同理,并非直接return子树,而是修剪过后再return
        root->left = traversal(root->left, L, R);//对符合区间条件的根节点左右子树进行操作
        root->right = traversal(root->right, L, R);
        return root;
    }

将有序数组转换为二叉搜索树

将一个按照升序排列的有序数组,转换为一棵高度平衡二叉搜索树。

本题中,一个高度平衡二叉树是指一个二叉树每个节点 的左右两个子树的高度差的绝对值不超过 1。

示例:

108.将有序数组转换为二叉搜索树

​ 按题目要求切割数组得到平衡二叉搜索树,则从数组中间作为根节点开始切割;

​ 同时注意循环不变量,因为这里是需要不断对数组进行切割直到子数组的长度为一;

​ 遍历代码如下:

	TreeNode* traversal(vector<int>& nums, int L, int R){//确定函数参数和返回值
        //这里选择左闭右开的切割区间,则子区间停止切割的终止条件即是数组长度等于零的时候
        if(L >= R)	return NULL;
        //单层递归逻辑
        int mid = (R-L)/2+L;//防止int溢出
        TreeNode* root = new TreeNode(nums[mid]);//确定根节点
        root->left = traversal(nums, L, mid);//左闭右开
        root->right = traversal(nums, mid+1, R);
        return root;
    }

把二叉搜索树转化为累加树

给出二叉 搜索 树的根节点,该树的节点值各不相同,请你将其转换为累加树(Greater Sum Tree),使每个节点 node 的新值等于原树中大于或等于 node.val 的值之和。

提醒一下,二叉搜索树满足下列约束条件:

节点的左子树仅包含键 小于 节点键的节点。 节点的右子树仅包含键 大于 节点键的节点。 左右子树也必须是二叉搜索树。

示例 1:

538.把二叉搜索树转换为累加树

  • 输入:[4,1,6,0,2,5,7,null,null,null,3,null,null,null,8]
  • 输出:[30,36,21,36,35,26,15,null,null,null,33,null,null,null,8]

示例 2:

  • 输入:root = [0,null,1]
  • 输出:[1,null,1]

示例 3:

  • 输入:root = [1,0,2]
  • 输出:[3,3,2]

示例 4:

  • 输入:root = [3,2,4,1]
  • 输出:[7,9,4,10]

提示:

  • 树中的节点数介于 0 和 104 之间。
  • 每个节点的值介于 -104 和 104 之间。
  • 树中的所有值互不相同
  • 给定的树为二叉搜索树。

​ 如果给定的是一个有序数组,则[1, 2, 3, 4]的结果就是[10, 9, 7, 4],那二叉树呢?

​ 从二叉搜索树的最大值开始往前递加,无疑就是处理顺序的改变,即右中左的顺序来遍历整个二叉搜索树,即反中序遍历;

​ 则递归代码如下:

	int sum = 0;//记录前一个节点的数值
	void traversal(TreeNode* cur){//确定函数参数和返回值:遍历整棵树且无须对返回值进行处理
        if(!cur)	return;//确定终止条件:为空返回
        //if(cur->right)	sum += cur->right->val;	单层逻辑
        traversal(cur->right);
        cur->val += sum;
        sum = cur->val;//更新sum
        //if(cur->left)	sum += cur->left->val;
        traversal(cur->left);
    }

​ 整体代码如下:

/**
 * 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 {
private:
    int sum = 0;//记录前一个节点的数值
	void traversal(TreeNode* cur){//确定函数参数和返回值:遍历整棵树且无须对返回值进行处理
        if(!cur)	return;//确定终止条件:为空返回
        //if(cur->right)	sum += cur->right->val;	单层逻辑
        traversal(cur->right);
        cur->val += sum;
        sum = cur->val;//更新sum
        //if(cur->left)	sum += cur->left->val;
        traversal(cur->left);
    }
public:
    TreeNode* convertBST(TreeNode* root) {
        sum = 0;
        traversal(root);
        return root;
    }
};

总结

​ 递归和迭代的思想都要熟悉;

递归

​ 首先是熟知递归函数的参数和返回值:

​ 如何利用才能使算法的性能最优化,例如对返回节点类型、传入int替代vector;

​ (见Day 18)

​ 如果需要搜索整棵二叉树且不用处理递归返回值,递归函数就不要返回值;(这种情况就是本文下半部分介绍的113.路径总和ii)

​ 如果需要搜索整棵二叉树且需要处理递归返回值,递归函数就需要返回值; (这种情况我们在236. 二叉树的最近公共祖先 (opens new window)中介绍)

​ 如果要搜索其中一条符合条件的路径,那么递归一定需要返回值,因为遇到符合条件的路径了就要及时返回;

​ 其次是关于递归函数的终止条件:

​ 有时候终止条件并不是简单的对处理节点判空即可,需要根据具体情境进行判断;

​ 最后是递归函数的单层遍历逻辑:

​ 二叉树题目的逻辑往往不算很难,大多时候可以理解为对一个遍历顺序特殊的数组进行处理,所以遍历顺序尤为重要;

​ 在构建二叉树的节点的时候,一定是选择从根节点开始(中左右),即前序遍历;

​ 在面对二叉搜索树时,一定是选择中序遍历(左中右),这样才能充分利用二叉搜索树的有序性质;

​ 求普通二叉树的属性的时候,一般采取后序遍历(左右中),因为需要返回中节点进行处理;

迭代

​ 迭代最主要的思想就是用栈来模拟树的逻辑,通过不断的push pop得到理想中的出栈顺序;

​ 一般在涉及到递归不是很好处理返回值的时候使用迭代,层序遍历就是一个典型的例子。

  • 29
    点赞
  • 18
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
要将一个以`.txt`格式存储的`yyyymmdd`格式的日期文件转换为对应的年份的第几天(day of year),你可以读取文件中的日期,并使用`datetime`函数创建日期时间对象,并使用`day`函数获取年份中的第几天。以下是一个示例代码: ```matlab filename = 'dates.txt'; % 输入的日期文件名 fileID = fopen(filename, 'r'); % 打开日期文件 dates = textscan(fileID, '%s'); % 读取日期文件中的所有日期 dates = dates{1}; % 将读取结果转换为列向量 fclose(fileID); % 关闭日期文件 n = length(dates); % 获取日期数量 dayOfYear = zeros(n, 1); % 初始化day of year向量 for i = 1:n dateStr = dates{i}; dt = datetime(dateStr, 'InputFormat', 'yyyyMMdd'); % 创建日期时间对象 dayOfYear(i) = day(dt, 'dayofyear'); % 获取年份中的第几天 end ``` 在这个例子中,我们首先定义了输入的日期文件名`filename`,然后使用`fopen`函数打开该文件。接下来,使用`textscan`函数读取日期文件中的所有日期,并将读取结果存储在`dates`变量中。之后,关闭日期文件。 然后,我们获取日期数量,并初始化一个与日期数量相同的零向量`dayOfYear`来保存day of year的结果。通过循环遍历日期向量的每个元素,使用`datetime`函数创建日期时间对象,并将日期字符串解析为对应的日期。最后,使用`day`函数来获取该日期时间对象的年份中的第几天,并将结果存储在`dayOfYear`向量中。 希望这个回答对你有帮助!如果还有其他问题,请随时提出。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值