代码随想录刷题——二叉树篇(十三)

106. 从中序与后序遍历序列构造二叉树
递归法(vector版本):

cpp 复制代码
class Solution{
public:
	TreeNode* traversal(vector<int>& inorder,vector<int> postorder){
		if(postorder.size()==0) return NULL;
		int rootValue = postorder.back();
		TreeNode* root = new TreeNode(rootValue);
		if(postorder.size()==1) return root;
		int i;
		for(i=0;i<inorder.size();i++){
			if(inorder[i]==rootValue) break;
		}
		vector<int> leftInorder(inorder.begin(),inorder.begin()+i);
		vector<int> rightInorder(inorder.begin()+i+1,inorder.end());
		postorder.pop_back();
		vector<int> leftPostorder(postorder.begin(),postorder.begin()+leftInorder.size());
		vector<int> rightPostorder(postorder.begin()+leftInorder.size(),postorder.end());
		root->left = traversal(leftInorder,leftPostorder);
		root->right = traversal(rightInorder,rightPostorder);
		return root;
	}
	TreeNode* buildTree(vector<int>& inorder,vector<int> postorder){
		if(inorder.size()==0||postorder.size()==0) return NULL;
		return traversal(inorder,postorder);
	}
};

递归法(下标版本):

cpp 复制代码
class Solution{
public:
	TreeNode* traversal(vector<int>& inorder,vector<int>& postorder,int inbegin,int inend,int postbegin,int postend){
		if(postbegin==psotend) return NULL;
		int rootValue=postorder[postend-1];
		TreeNode* root=new TreeNode(rootValue);
		if(postend-postbegin==1) return root;
		int i;
		for(i=inbegin;i<inend;i++){
			if(inorder[i]==rootValue) break;
		} 
		int linbegin = inbegin;
		int linend = i; 
		int rinbegin = i+1;
		int rinend = inend;
		
		int lpostbegin = postbegin;
		int lpostend = postbegin + i - inbegin;
		int rpostbegin = postbegin + i - inbegin;
		int rpostend = postend - 1;
		root->left = traversal(inorder,postorder,linbegin,linend,lpostbegin,lpostend);
		root->right = traversal(inorder,postorder,rinbegin,rinend,rpostbegin,rpostend);
		return root;
	}
	TreeNode* buildTree(vector<int> inorder,vector<int> postorder){
		if(inorder.size()==0||postorder.size()==0) return NULL;
		return traversal(inorder,postorder,0,inorder.size(),0,postorder.size());
	}
};

105. 从前序与中序遍历序列构造二叉树
递归法(下标版本):

cpp 复制代码
class Solution{
public:
	TreeNode* traversal(vector<int>& preorder,vector<int>& inorder,int inbegin,int inend,int prebegin,int preend){
		if(prebegin==preend) return NULL;
		TreeNode* root = new TreeNode(preorder[prebegin]);
		if(preend-prebegin==1) return root;
		int i;
		for(i=inbegin;i<inend;i++){
			if(inorder[i]==preorder[prebegin]) break;
		}
		int linbegin=inbegin;
		int linend=i;
		prebegin++;
		int lprebegin=prebegin;
		int lpreend=prebegin+i-inbegin;
		
		int rinbegin=i+1;
		int rinend=inend;
		int rprebegin=prebegin+i-inbegin;
		int rpreend=preend;
		root->left=traversal(preorder,inorder,linbegin,linend,lprebegin,lpreend);
		root->right=traversal(preorder,inorder,rinbegin,rinend,rprebegin,rpreend);
		return root;
	}
	TreeNode* buildTree(vector<int>& preorder, vector<int>& inorder) {
        if(preorder.size()==0||inorder.size()==0) return NULL;
        return traversal(preorder,inorder,0,preorder.size(),0,inorder.size());
    }
};

其他:

(1)先找根节点:后序遍历最后一个 ,然后用后序遍历的节点切中序遍历 ,再根据中序遍历切开的左右子数组切后序遍历。

这张图画的太好了:

vector的一些操作:back()、find()、resize()、pop_back()

(2)vector<int> vec(inorder.begin(),inorder.begin()+num)这种构建方式,num=0时会构建一个空的vector,构造时可以按意义理解:vec(i,j) ,则取**[ i , j )** ,左闭右开 ,迭代器可以理解成指向一个位置的指针 ,加几移动几个位置

(3)这道题的递归法其实和之前的思路一样,只不过是把inorderpostorder 这两个遍历数组的不同情况节点所处的位置类型进行了对应:

a.inorder和postorder为 ------------------------>到了空节点

b.inorder和postorder的size为1 ----------------->到了叶子节点

c.其他(递归函数的其他区域)---------------->其他节点处理逻辑

和三部曲基本一致,先确定参数 (中序和后序数组),再确定终止处理条件 (到空节点和叶子节点(这里之所以是两个情况,是因为存在某个节点只有一个子节点的情况)),最后确定其他节点的处理逻辑(切分数组、递归)

下标版也是一样:

a.postbegin**==** postend----------------------------->到了空节点

b.postend**-** postbegin**==0** --------------------------->到了叶子节点

c.其他(递归函数的其他区域)------------------>其他节点处理逻辑

(4)下标版本同样需要保持左闭右开 的状态,下标左为闭区间右为开区间

(5)105题一开始traversal函数的vector<int>参数忘记写引用 了,执行用时和消耗内存都差了很多,原因是不加引用的话,每次递归都会把vector<int>参数复制一遍

(6)中序+前序/后序数组可以构建二叉树,只有前序和后序不行

相关推荐
先吃饱再说6 小时前
判断回文字符串,从一行代码到双指针优化
算法
黄敬峰8 小时前
深入理解算法核心:从递归思想、数组扁平化到快速排序
算法
得物技术10 小时前
从狂野代码到按目标生产:得物推荐 AI Harness 的工程化实践|AICon 演讲整理
人工智能·算法·架构
AI小老六13 小时前
SkillOpt 架构拆解:把 Skill 文本当参数,用执行轨迹训练 Agent
后端·算法·ai编程
胡萝卜术14 小时前
从“分数打架”到“排名投票”:为什么你的ChatBI必须用RRF?
算法·设计模式·面试
Asize14 小时前
初识DFS 与 BFS:递归、队列与图遍历
算法
罗西的思考1 天前
机器人 / 强化学习】HIL-SERL:人类在环驱动的具身智能进化框架
人工智能·算法·机器学习
CSharp精选营1 天前
关系型 vs 非关系型:从原理到选型,一文搞定数据库核心分类
数据结构·nosql·关系型数据库·非关系型数据库·技术选型
美团技术团队1 天前
LongCat 开源 VitaBench 2.0:长期动态智能体基准新标杆
人工智能·算法