算法day16|654.最大二叉树、617.合并二叉树、700.二叉搜索树中的搜索、98.验证二叉搜索树

news/2025/1/15 18:28:21/

算法day16|654.最大二叉树、617.合并二叉树、700.二叉搜索树中的搜索、98.验证二叉搜索树

  • 654.最大二叉树
  • 617.合并二叉树
    • 1.额外申请空间(失败)
    • 2.不额外申请空间
  • 700.二叉搜索树中的搜索
  • 98.验证二叉搜索树
    • 1.遍历后排序
    • 2.边遍历遍排序
    • 3.指针记录法

654.最大二叉树

这道题很简单,其实就是105、106的变式题。具体代码如下:

class Solution {
public:TreeNode*traversal(vector<int>& nums){if(nums.empty())return nullptr;int max=nums[0];int index=0;for(int i=1;i<nums.size();i++){if(nums[i]>max){max=nums[i];index=i;} }TreeNode*root=new TreeNode(max);if(nums.size()==1)return root;vector<int> leftNums(nums.begin(),nums.begin()+index);vector<int> rightNums(nums.begin()+index+1,nums.end());root->left=traversal(leftNums);root->right=traversal(rightNums);return root;}TreeNode* constructMaximumBinaryTree(vector<int>& nums) {if(!nums.empty())return traversal(nums);elsereturn nullptr;}
};

总体思路与105、106类似,甚至更简单。

617.合并二叉树

1.额外申请空间(失败)

不知道为什么运行不了…,代码如下:

class Solution {
public:TreeNode* traversal(TreeNode* root1, TreeNode* root2){if(root1==nullptr&&root2==nullptr)return nullptr;TreeNode*root=new TreeNode();if(root1!=nullptr&&root2!=nullptr)root->val=root1->val+root2->val;else if(root1==nullptr&&root2!=nullptr)root->val=root2->val;else if(root1!=nullptr&&root2==nullptr)root->val=root1->val;root->left=traversal(root1->left,root2->left);root->right=traversal(root1->right,root2->right);return root;}TreeNode* mergeTrees(TreeNode* root1, TreeNode* root2) {return traversal(root1,root2);}
};

2.不额外申请空间

class Solution {
public:
TreeNode* traversal(TreeNode* root1, TreeNode* root2){if(!root1&&!root2)return nullptr;else if(!root1)return root2;else if(!root2)return root1;else{root1->val+=root2->val;}root1->left=traversal(root1->left,root2->left);root1->right=traversal(root1->right,root2->right);return root1;}TreeNode* mergeTrees(TreeNode* root1, TreeNode* root2) {return traversal(root1,root2);}
};

直接在root1上进行操作,不用额外申请空间。

700.二叉搜索树中的搜索

class Solution {
public:TreeNode* traversal(TreeNode* root, int val){if(root==nullptr)return nullptr;if(root->val>val)return traversal(root->left,val);if(root->val<val)return traversal(root->right,val);elsereturn root;}TreeNode* searchBST(TreeNode* root, int val) {return traversal(root,val);}
};

要注意一下BST的特点:BST首先得是二叉平衡树,满足左<中<右。所以:

if(root->val>val)return traversal(root->left,val);if(root->val<val)return traversal(root->right,val);

另外,如果递归有返回值的话,在单层递归里面肯定是需要设置变量来接收的,或者直接return 递归。

98.验证二叉搜索树

1.遍历后排序

class Solution {
public:void traversal(TreeNode* root,vector<int> &vec){if(root==nullptr)return ;traversal(root->left,vec);vec.push_back(root->val);traversal(root->right,vec);}bool isValidBST(TreeNode* root) {if(root==nullptr)return true;else{vector<int> vec;traversal(root,vec);for(int i=0;i<vec.size()-1;i++){if(vec[i]>=vec[i+1])return false;}return true;}}
};

这题的易错点就是必须保证左子树上的所有元素都要小于根节点,右子树同理,而不是仅仅是单个左孩子结点或者右孩子结点。这样的思路用递归就很难实现了。

所以我们另辟蹊径,利用二叉搜索树的最重要的特征之一:中序序列单调递增 。我们只需要用数组收集中序序列,然后去判断它是否递增即可。

2.边遍历遍排序

对于递增的判断其实是可以在遍历过程中就实现的,代码如下:

class Solution {
public:long long MaxValue = LONG_MIN;bool traversal(TreeNode* root){if(root==nullptr)return true;bool left=traversal(root->left);if(root->val>MaxValue)MaxValue=root->val;elsereturn false; bool right=traversal(root->right);return left&&right;}bool isValidBST(TreeNode* root) {return traversal(root);}
};

为了判断中序序列是否递增,我们需要一个值来继承之前结点的大小,然后与当前结点比较即可。由于力扣中输入了Int的最小值,所以我们采用long long型的最小值来接受第一个元素,十分巧妙:

		long long MaxValue = LONG_MIN;if(root->val>MaxValue)MaxValue=root->val;elsereturn false;

这样就可以保证第一个数的顺利进行。因为不想for循环,递归的时候是很难定位到第几个元素的,所以想把第哪个元素赋值为几,这是做不到的。只有在深刻理解逻辑之后做一些巧思。

3.指针记录法

class Solution {
public:TreeNode*pre=nullptr;bool traversal(TreeNode* root){if(root==nullptr)return true;bool left=traversal(root->left);if(pre!=nullptr&&pre->val>=root->val)return false;elsepre=root;bool right=traversal(root->right);return left&&right;}bool isValidBST(TreeNode* root) {return traversal(root);}
};

我们改用指针来记录,实现一个更巧妙的逻辑:用指针记录前面的值,只有当指针的值大于当前的值时,return false。问题在于,指针该如何记录呢?逻辑很巧妙,因为记录第一个值是非常关键的。初始时,我们设pre为null,第一次就是因为pre为null,成功赋值;其他时候是因为满足排序,所以成功赋值。它们的逻辑是有区别的。


http://www.ppmy.cn/news/1521828.html

相关文章

C语言——插入排序

先将序列的第1个记录看成是一个有序的子序列&#xff0c;然后从第2个记录逐个进行插入&#xff0c;直至整个序列有序为止。 #include <stdio.h> #include <stdlib.h> void insertion_sort(int *arr, int n) { for (int i 1; i < n; i) { int …

Android 11添加系统服务,并封装jar包供第三方应用使用

概述&#xff1a; 如果你是做技术支持&#xff0c;有没有遇到这种情况&#xff0c;客户既要实现具备系统权限的功能&#xff0c;但是呢&#xff0c;又不想把自己的应用做成系统应用。这时候你咋办。 我们可以添加一个具备系统权限的服务&#xff0c;不管前台的&#xff0c;还是…

Linux 常用命令 - hexdump 【以指定格式显示文件内容】

简介 hexdump 可以将指定文件或标准输入按照指定的格式进行输出&#xff0c;其可以用来查看任何文件的原始数据&#xff0c;在分析非文本文件的场景下非常有用。 使用方式 hexdump [-bcCdovx] [-e 指定格式] [-f 指定文件] [-n 长度] [-s 偏移] file ... hd [-bcdovx] [-e 指…

Leetcode每日刷题之155.最小栈

1.题目解析 本题是实现一个栈并且要实现其中的插入、删除、返回栈顶元素、返回最小元素的函数&#xff0c;这里主要的难点就是返回最小元素的函数&#xff0c;如果我们直接遍历&#xff0c;那么时间复杂度就是O(N)&#xff0c;但是题目要求我们需要在常数时间也就是O(1)的时间复…

Docker必备命令集合,让你轻松驾驭容器化

Docker作为现代化应用程序的部署和管理平台&#xff0c;已经成为开发者和运维工程师的得力工具。但对于新手而言&#xff0c;面对众多的命令和参数&#xff0c;有时会感到困惑。本文将为你总结一组常用的Docker命令&#xff0c;助你快速上手并高效使用这一强大工具。 1. 基础命…

Qt数字化信息通讯调制解调

对于数字化信息通讯调制解调&#xff0c;Qt本身并不直接提供调制解调的功能&#xff0c;但是可以通过Qt的网络编程接口&#xff0c;结合相关的算法和硬件设备来实现。例如&#xff0c;可以通过Qt的信号处理库来实现数字信号的调制和解调算法&#xff0c;或者通过串口通信与外部…

word中怎么快速选中光标之前或之后的全部内容?

在Word中&#xff0c;快速选中光标之后的全部内容的快捷键&#xff1a;Ctrl Shift End&#xff1b; 在Word中&#xff0c;快速选中光标之前的全部内容的快捷键&#xff1a;Ctrl Shift Home。 在Word中&#xff0c;选取的快捷键如下。 一、选定整个文本&#xff1a; 1&#…

微信小程序认证和备案

小程序备案的流程一般包括以下步骤‌&#xff1a; 准备备案所需材料‌&#xff1a;通常需要提供‌营业执照、法人的‌身份证、两个‌手机号和一个邮箱等资料。 ‌1 ‌登录‌微信公众平台‌&#xff1a;作为第一次开发微信小程序的服务商&#xff0c;需要通过微信公众平台申请…