数据结构初阶--二叉树的链式结构

这篇具有很好参考价值的文章主要介绍了数据结构初阶--二叉树的链式结构。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

目录

一.二叉树链式结构的概念

二.二叉树链式结构的功能实现

2.1.链式二叉树的定义

2.2.链式二叉树的构建

2.3.链式二叉树的遍历

2.3.1.先序遍历

2.3.2.中序遍历

2.3.3.后序遍历

2.3.4.层序遍历

2.4.链式二叉树的求二叉树的结点数量

法一:计数法

法二:分治法

2.5.链式二叉树的求二叉树的叶子结点数量

2.6.链式二叉树的求二叉树第k层结点的数量

2.7.链式二叉树的求二叉树的高度

2.8.链式二叉树的查找二叉树中值为x的结点

2.9.链式二叉树的判断二叉树是否是完全二叉树

2.10.链式二叉树的二叉树的销毁

三.二叉树基础OJ练习

题一:单值二叉树

题二:相同的树

题三:对称二叉树

题四:另一棵树的子树

题五:二叉树的前序遍历

题六:二叉树遍历


前置说明:

在学习二叉树的基本操作之前,需要先创建一棵二叉树,然后才能学习其相关的基本操作。由于现在对于二叉树的结构掌握还不够深,为了降低大家的学习成本,此处手动创建一棵简单的二叉树,以便快速进入二叉树的操作学习。等二叉树结构了解的差不多时,我们反过头再来研究二叉树真正的创建方式。

一.二叉树链式结构的概念

对于任意的二叉树来说,每个结点只有一个双亲结点(根除外),最多有两个孩子。可以设计每个结点至少包括三个域:数据域data,左孩子域left和右孩子域right。其中,left域指向该结点的左孩子,data域记录该结点的信息,right域指向该结点的右孩子。此结点结构形成的二叉树称为二叉链表。

数据结构初阶--二叉树的链式结构,数据结构,数据结构

二.二叉树链式结构的功能实现

若有n个结点,则有2n个指针域,而除了根结点每个结点头上都会连一个指针,故n个结点的二叉链表共有n+1个空链域。

2.1.链式二叉树的定义

typedef int BTDataType;

typedef struct BinaryTreeNode
{
	struct BinaryTreeNode* left;//左孩子指针
	struct BinaryTreeNode* right;//右孩子指针
	BTDataType data;//数据域
}BTNode;

2.2.链式二叉树的构建

//创建结点
BTNode* BuyNode(BTDataType x)
{
	BTNode* node = (BTNode*)malloc(sizeof(BTNode));
	assert(node);

	node->data = x;
	node->left = NULL;
	node->right = NULL;

	return node;
}

//构造二叉树
BTNode* CreatBinaryTree()
{
	BTNode* node1 = BuyNode(1);

	BTNode* node2 = BuyNode(2);

	BTNode* node3 = BuyNode(3);

	BTNode* node4 = BuyNode(4);

	BTNode* node5 = BuyNode(5);

	BTNode* node6 = BuyNode(6);

	node1->left = node2;
	node1->right = node4;
	node2->left = node3;
	node4->left = node5;
	node4->right = node6;

	return node1;
}

调试分析:

数据结构初阶--二叉树的链式结构,数据结构,数据结构

运行结果:

数据结构初阶--二叉树的链式结构,数据结构,数据结构

注意:

这种方式可以很简单地找到指定结点的左右孩子,但是要找到指定结点的父结点就只能从根结点开始遍历寻找。

2.3.链式二叉树的遍历

二叉树的遍历是指按一定规律对二叉树中的每个结点进行访问且仅访问一次。二叉树是非线性数据结构,通过遍历可以将二叉树中的结点访问一次且仅一次,从而得到访问结点的顺序序列。从这个意义上说,遍历操作就是将二叉树中结点按一定规律线性化的操作,目的在于将非线性化结构变成线性化的访问序列。

用L,D,R分别表示遍历左子树,访问根结点,遍历右子树。如果规定按先左后右的顺序,根据对根的访问先后顺序的不同,可以将DLR称为先序遍历或先根遍历,将LDR称为中序遍历,将LRD称为后序遍历。

注意:先序,中序和后序遍历都是递归定义的,即在其子树中亦按上述规律进行遍历。

案例分析:

数据结构初阶--二叉树的链式结构,数据结构,数据结构

2.3.1.先序遍历

若二叉树为空,则什么也不做;否则依次执行如下三个操作:

  1. 访问根结点;
  2. 先序遍历左子树;
  3. 先序遍历右子树。 

数据结构初阶--二叉树的链式结构,数据结构,数据结构

实现:

void PreOrder(BTNode* root)
{
	//若树为空,则退出
	if (root == NULL)
	{
		printf("# ");//若结点为空,则用#表示
		return;
	}

	//先访问根结点
	printf("%d ", root->data);
	
	//然后访问左子树
	PreOrder(root->left);

	//再访问右子树
	PreOrder(root->right);
}

运行结果:

数据结构初阶--二叉树的链式结构,数据结构,数据结构

递归分析:

数据结构初阶--二叉树的链式结构,数据结构,数据结构

2.3.2.中序遍历

若二叉树为空,则什么也不做;否则依次执行如下三个操作:

  1. 中序遍历左子树;
  2. 访问根结点;
  3. 中序遍历右子树。

实现:

void InOrder(BTNode* root)
{
	//若树为空,则退出
	if (root == NULL)
	{
		printf("# ");//若结点为空,则用#表示
		return;
	}

	//先访问左子树
	InOrder(root->left);

	//然后访问根结点
	printf("%d ",root->data);

	//再访问右子树
	InOrder(root->right);
}

运行结果:

数据结构初阶--二叉树的链式结构,数据结构,数据结构

递归分析:

数据结构初阶--二叉树的链式结构,数据结构,数据结构

2.3.3.后序遍历

若二叉树为空,则什么也不做;否则依次执行如下三个操作:

  1. 后序遍历左子树;
  2. 后续遍历右子树;
  3. 访问根结点。

实现:

void PostOrder(BTNode* root)
{
	//若树为空,则退出
	if (root == NULL)
	{
		printf("# ");//若结点为空,则用#表示
		return;
	}

	//先访问左子树
	PostOrder(root->left);

	//然后访问右子树
	PostOrder(root->right);

	//再访问根结点
	printf("%d ", root->data);
}

运行结果:

数据结构初阶--二叉树的链式结构,数据结构,数据结构

小结:

递归算法的时间复杂度分析:设二叉树有n个结点,对每个结点都要进行一次入栈和出栈的操作,即入栈和出栈各执行n次,对结点的访问也是n次。这些二叉树递归遍历算法的时间复杂度为0(n)。

2.3.4.层序遍历

设二叉树的根结点所在层数为1,层序遍历就是从所在二叉树的根结点出发,首先访问第一层的树的根结点,然后从左到右访问第二层的结点,接着是第三层的结点,以此类推,自上而下,自左至右逐层访问树的结点的过程就是层序遍历。

实现方式:采用辅助队列实现按层序遍历二叉树。

数据结构初阶--二叉树的链式结构,数据结构,数据结构

算法思想:

  1. 初始化一个辅助队列;
  2. 根结点入队;
  3. 若队列非空,则队头结点出队,访问该结点,并将其左,右孩子插入队尾(如果有的话);
  4. 重复第三步直至队列为空。

实现:

void LevelOrder(BTNode* root)
{
	//创建队列并初始化
	Queue q;
	QueueInit(&q);

	//若根结点不为空,则先将根结点入队
	if (root)
	{
		//入队
		QueuePush(&q, root);
	}

	//若队列非空
	while (!QueueEmpty(&q))
	{
		//取队头元素
		BTNode* front = QueueFront(&q);
		//访问队头结点
		printf("%d ",front->data);
		//出队
		QueuePop(&q);

		//若左结点存在,则左结点入队
		if (front->left)
		{
			//入队
			QueuePush(&q, front->left);
		}

		//若右节点存在,则右结点入队
		if (front->right)
		{
			//入队
			QueuePush(&q, front->right);
		}
	}

	printf("\n");

	//销毁队列
	QueueDestory(&q);
}

运行结果:

数据结构初阶--二叉树的链式结构,数据结构,数据结构

2.4.链式二叉树的求二叉树的结点数量

法一:计数法

在递归调用过程中,局部变量和全局变量的传递方式可以类似的看成函数调用时的值传递和引用传递。

局部变量:类似于值传递过程中的形参和实参,可以看成是两个完全不同的值,只是名字相同而已。每次递归调用时都会重新创建新的变量,它并不会覆盖掉上次递归调用过程中创建的值。

全局变量:类似于传地址过程中的形参和实参,可以看成是同一个变量。每次递归调用时并不会重新创建新的变量,在递归过程中对全局变量的修改,会影响到下一次递归调用过程中的值。

所以,我们在使用计数器进行结点个数统计是,要定义全局变量而非局部变量。

当使用局部变量时:

int TreeSize(BTNode* root)
{
	int count = 0;

	//若树为空
	if (root == NULL)
	{
		return;
	}

	//统计根结点
	++count;

	//统计左子树
	TreeSize(root->left);

	//统计右子树
	TreeSize(root->right);

	return count;
}

运行结果:

数据结构初阶--二叉树的链式结构,数据结构,数据结构

当使用全局变量时:

int count = 0;//不能定义一个局部变量,因为每次递归调用都会重新初始化为0

void TreeSize(BTNode* root)
{
	//若树为空
	if (root == NULL)
	{
		return;
	}

	//统计根结点
	++count;

	//统计左子树
	TreeSize(root->left);

	//统计右子树
	TreeSize(root->right);

}

运行结果:

数据结构初阶--二叉树的链式结构,数据结构,数据结构

这里存在一个问题:当我们多次打印输出结果时,可以发现每次的输出结果并不相同。如下所示:

数据结构初阶--二叉树的链式结构,数据结构,数据结构

为了避免该类问题的出现,只需在每次调用之前要将count进行初始化,防止下次调用时将前一次的调用结果进行累加。如下所示:

数据结构初阶--二叉树的链式结构,数据结构,数据结构

法二:分治法

使用计数法计算结点个数时存在较多需要注意的问题,而使用分治法可以有效解决此类问题。

采用递归算法,首先判断树是否为空树,若为空树,则返回0;若不是空树,则将根结点root的左孩子结点和右孩子结点作为参数传给函数TreeSize进行递归调用。

实现:

int TreeSize(BTNode* root)
{
	return root == NULL ? 0 : TreeSize(root->left) + TreeSize(root->right) + 1;
}

递归分析:

数据结构初阶--二叉树的链式结构,数据结构,数据结构

运行结果:

数据结构初阶--二叉树的链式结构,数据结构,数据结构

2.5.链式二叉树的求二叉树的叶子结点数量

采用递归算法,如果是空树,返回0;如果只有一个结点,返回1;否则为左右子树的叶子结点数之和。

实现:

int TreeLeafSize(BTNode* root)
{
	//若树为空
	if (root == NULL)
	{
		return 0;
	}

	//若为单独的叶子,即只含一个结点
	if (root->left == NULL && root->right == NULL) 
	{
		return 1;
	}

	//若树不为空,也不为单独的叶子
	return TreeLeafSize(root->left) + TreeLeafSize(root->right);
}

运行结果:

数据结构初阶--二叉树的链式结构,数据结构,数据结构

2.6.链式二叉树的求二叉树第k层结点的数量

采用递归算法,首先判断k的取值是否合法,k的取值不能小于1,然后判断树是否为空,如果是空树,则返回0;如果树不为空,且k的值为1,则返回1;否则为左右子树的第k-1层的结点数之和。

实现:

int TreeKLevel(BTNode* root, int k)
{
	//检查k的范围
	assert(k >= 1);

	//若树为空
	if (root == NULL)
	{
		return 0;
	}

	//若树不为空,且k等于1,也就是根结点所在层次
	if (k == 1)
	{
		return 1;
	}

	//求左子树和右子树的第k-1层的结点数之和
	return TreeKLevel(root->left, k - 1) + TreeKLevel(root->right, k - 1);
}

递归分析:

数据结构初阶--二叉树的链式结构,数据结构,数据结构

运行结果:

数据结构初阶--二叉树的链式结构,数据结构,数据结构

2.7.链式二叉树的求二叉树的高度

采用递归算法,首先判断树是否为空,若为空,则返回0;若不为空,则递归求左子树的高度和右子树的高度,然后再求出左子树和右子树高度的较大值,最后再将高度的最大值+1,即为所求二叉树的高度。

实现:

int TreeDepth(BTNode* root)
{
	//若树为空
	if (root == NULL)
	{
		return 0;
	}

	//求左树的高度
	int leftDepth = TreeDepth(root->left);

	//求右树的高度
	int rightDepth = TreeDepth(root->right);

	//求两者中较大值
	return leftDepth > rightDepth ? leftDepth + 1 : rightDepth + 1;
}

运行结果:

数据结构初阶--二叉树的链式结构,数据结构,数据结构

2.8.链式二叉树的查找二叉树中值为x的结点

采用递归算法,首先判断树是否为空,若树为空,则返回NULL。当树不为空,且根结点为所要查找的结点时,则返回根结点。否则,递归遍历左子树和右子树。

实现:

BTNode* TreeFind(BTNode* root, BTDataType x)
{
	//若树为空
	if (root == NULL)
	{
		return NULL;
	}

	//若根结点为所要查找的结点
	if (root->data == x)
	{
		return root;
	}

	//查找左子树
	BTNode* ret1 = TreeFind(root->left, x);

	//若返回值不为空
	if (ret1)
	{
		return ret1;
	}

	//查找右子树
	BTNode* ret2 = TreeFind(root->right, x);

	//若返回值不为空
	if (ret2)
	{
		return ret2;
	}

	return NULL;
}

递归分析:

数据结构初阶--二叉树的链式结构,数据结构,数据结构

运行结果:

数据结构初阶--二叉树的链式结构,数据结构,数据结构

2.9.链式二叉树的判断二叉树是否是完全二叉树

完全二叉树:对于一棵深度为h的二叉树,其前h-1层结点个数均达到最大值,第h层结点个数可能没有达到最大值,但从左到右是连续的。

算法思想:

  1. 初始化一个辅助队列;
  2. 根结点入队;
  3. 若队列非空,则队头结点出队,访问该结点,并将其左,右孩子插入队尾(包含空结点);
  4. 重复第三步直至遇到空结点,并查看其后续是否有非空结点。若有,则该二叉树不是完全二叉树;若没有,则该二叉树是完全二叉树。

实现:

int TreeComplete(BTNode* root)
{
	//初始化队列
	Queue q;
	QueueInit(&q);

	//先将根结点入队
	if (root)
	{
		QueuePush(&q, root);
	}

	while (!QueueEmpty(&q))
	{
		//读取队头元素并出队
		BTNode* front = QueueFront(&q);
		QueuePop(&q);

		//若队头结点非空
		if (front)
		{
			//左结点入队,包含空结点
			QueuePush(&q, front->left);

			//右结点入队,包含空结点
			QueuePush(&q, front->right);
		}
		else
		{
			//遇到空,则跳出层序遍历
			break;
		}
	}

	//1.如果空后面全是空结点,则是完全二叉树
	//2.如果空后面还有非空结点,则不是完全二叉树
	while (!QueueEmpty(&q))
	{
		//读取队头元素并出队
		BTNode* front = QueueFront(&q);
		QueuePop(&q);

		//front存放的是树的结点的指针
		if (front)
		{
			QueueDestory(&q);
			return false;
		}
	}

	//销毁
	QueueDestory(&q);

	return true;
}

运行结果:

数据结构初阶--二叉树的链式结构,数据结构,数据结构

2.10.链式二叉树的二叉树的销毁

采用递归算法,首先判断树是否为空,若为空,则返回NULL。若树不为空,则先递归销毁左子树,然后再销毁右子树,最后再释放根结点。

实现:

void TreeDestroy(BTNode* root)
{
	//若树为空
	if (root == NULL)
	{
		return;
	}

	//销毁左子树
	TreeDestroy(root->left);

	//销毁右子树
	TreeDestroy(root->right);

	printf("%p:%d\n", root, root->data);

	//释放根结点
	free(root);
}

运行结果:

数据结构初阶--二叉树的链式结构,数据结构,数据结构

三.二叉树基础OJ练习

题一:单值二叉树

题目描述:

如果二叉树每个结点都具有相同的值,那么该二叉树就是单值二叉树。只有给定的树是单值二叉树,才返回true;否则返回false。

法一:标记法

设置标志位flag进行单值二叉树的判断。若树为空或者标志位flag为flase,则直接返回;若根结点的值不等于val,则将标志位flag设为flase,同时返回;若上述两个条件均满足,则递归遍历左子树和右子树。

实现:

struct TreeNode
{
	int val;
	struct TreeNode* left;
	struct TreeNode* right;
};

//法一:设置标记位
bool flag = true;
void PreOrderCompare(struct TreeNode* root, int val)
{
	//若树为空
	if (root == NULL || flag == false)//flag==flase:若左子树不满足条件则直接退出,以免对右子树进行不必要的递归遍历
	{
		return;
	}

	//如果根结点的值不等于val,则直接退出,不用往下递归
	if (root->val != val)
	{
		flag = false;
		return;
	}
	
	//比较左子树
	PreOrderCompare(root->left, val);

	//比较右子树
	PreOrderCompare(root->right,val);
}

bool isUnivalTree(struct TreeNode* root)
{
	//若树为空
	if (root == NULL)
	{
		return true;
	}

	flag = true;

	PreOrderCompare(root, root->val);

	return flag;
}

法二:

首先判断根结点root是否为NULL,若为空则直接返回true。然后判断根结点root的左孩子与右孩子是否为空且其值与root的值是否相同,若不同,则返回false;若相同,则直接递归遍历root的左子树和右子树,判断其否为单值二叉树。

实现:

struct TreeNode
{
	int val;
	struct TreeNode* left;
	struct TreeNode* right;
};

bool isUnivalTree(struct TreeNode* root)
{
	//如果树为空
	if (root == NULL)
	{
		return true;
	}

	//若左孩子不为空且左孩子的值不等于根结点
	if (root->left && root->left->val != root->val)
	{
		return false;
	}

	//若右孩子不为空且右孩子的值不等于根结点
	if (root->right && root->right->val != root->val)
	{
		return false;
	}

	//递归遍历左子树和右子树
	return isUnivalTree(root->left) && isUnivalTree(root->right);//递归调用的返回,都是返回到递归调用的上一层
}

递归分析:

数据结构初阶--二叉树的链式结构,数据结构,数据结构

题二:相同的树

题目描述:

给你两棵二叉树的根结点p和q,编写一个函数来检验这两棵树是否相同。如果两棵树在结构上相同,并且结点具有相同的值,则认为它们是相同的。

分析:

首先判断两棵树是否均为空,若均为空,则返回true;若有一棵树为空,则返回false。然后在两棵树均不为空的情况下,判断其对应的根结点的值是否相等,若不相等则返回false。最后再递归遍历两棵树对应的左子树和右子树,判断其对应的结点值是否相等。

实现:

struct TreeNode
{
	int val;
	struct TreeNode* left;
	struct TreeNode* right;
};

bool isSameTree(struct TreeNode* p, struct TreeNode* q)
{
	//如果两棵树都为空
	if (p == NULL && q == NULL)
	{
		return true;
	}

	//如果两棵树有一棵为空
	if (p == NULL || q == NULL)
	{
		return false;
	}

	//如果两棵子树都不为空,且两个根结点的值不相等
	if (p->val != q->val)
	{
		return false;
	}

	return isSameTree(p->left, q->left) && isSameTree(p->right, q->right);
}

题三:对称二叉树

题目描述:

给你一个二叉树的根结点root,检查它是否轴对称。

分析:

首先判断树是否为空,若为空则返回true,若不唯空,则判断根结点root对应的左子树和右子树。然后转为对左子树和右子树的判断,若两棵子树均为空,咋返回true;若有一棵子树为空,则返回false;若两棵子树均不为空,且其对应的根结点的值不相等,则返回false。最后再递归遍历,判断左子树的左子树与右子树的右子树以及左子树的右子树与右子树的左子树是否相等。

实现:

struct TreeNode
{
	int val;
	struct TreeNode* left;
	struct TreeNode* right;
};

bool isSymmetricSubTree(struct TreeNode* root1, struct TreeNode* root2)
{
	//若两棵树均为空
	if (root1 == NULL && root2 == NULL)
	{
		return true;
	}

	//若有一棵树为空
	if (root1 == NULL || root2 == NULL)
	{
		return false;
	}

	//若两棵树均不为空
	if (root1->val != root2->val)
	{
		return false;
	}

	return isSymmetricSubTree(root1->left, root2->right) && isSymmetricSubTree(root1->right, root2->left);
}

bool isSymmetric(struct TreeNode* root)
{
	//若树为空
	if (root == NULL)
	{
		return true;
	}

	//若树不为空
	return isSymmetricSubTree(root->left, root->right);
}

题四:另一棵树的子树

题目描述:

给你两棵二叉树root和subRoot。检验root中是否包含和subRoot具有相同结构和节点值的子树。如果存在,返回true;否则,返回false。
二叉树tree的一棵子树包括tree的某个节点和这个节点的所有后代节点。tree也可以看做它自身的一棵子树。

分析:

遍历二叉树root中的每一个结点,然后将每一个结点都作为根结点并与subRoot的根结点进行比较,若对应的根结点值相同,则调用函数isSameTree进行左右子树的比较。如果左右子树均相同,则subRoot是root的子树;否则继续将二叉树root的下一个结点与subRoot的根结点进行比较。重复上述操作,直至找到一棵子树与二叉树subRoot相同或者二叉树root的所有子树均与subRoot不同并退出。

实现:

struct TreeNode
{
	int val;
	struct TreeNode* left;
	struct TreeNode* right;
};

bool isSameTree(struct TreeNode* p, struct TreeNode* q)
{
	//如果两棵树都为空
	if (p == NULL && q == NULL)
	{
		return true;
	}

	//如果两棵树有一棵为空
	if (p == NULL || q == NULL)
	{
		return false;
	}

	//如果两棵子树都不为空,且两个根结点的值不相等
	if (p->val != q->val)
	{
		return false;
	}

	return isSameTree(p->left, q->left) && isSameTree(p->right, q->right);
}

bool isSubtree(struct TreeNode* root, struct TreeNode* subRoot)
{
	//若树为空
	if (root == NULL)
		return false;

	//若树不为空
	if (isSameTree(root, subRoot))
	{
		return true;
	}

	//将原树中所有子树都找出来,然后跟subRoot比较
	return isSubtree(root->left, subRoot) || isSubtree(root->right, subRoot);
}

递归分析:

数据结构初阶--二叉树的链式结构,数据结构,数据结构

题五:二叉树的前序遍历

题目描述:

给你二叉树的根结点root,返回它结点值的前序遍历。

分析:

实现函数TreeSize,用以统计二叉树中的结点个数并返回给returnSize。同时开辟好returnSize个int 类型大小的数组空间,用于存放遍历后的各个结点。最后再调用函数preorder进行先序遍历,并返回数组首元素的地址。

实现:

struct TreeNode
{
	int val;
	struct TreeNode* left;
	struct TreeNode* right;
};

//求二叉树的结点个数
int TreeSize(struct TreeNode* root)
{
	return root == NULL ? 0 : TreeSize(root->left) + TreeSize(root->right) + 1;
}

//前序遍历,并将遍历后的结点值存入数组中
void preorder(struct TreeNode* root, int* a, int* pi)//pi的值要加引用,否则形参的修改不会影响实参
{
	//如果树为空
	if (root == NULL)
	{
		return;
	}

	//若树不为空
	a[(*pi)++] = root->val;

	//遍历左子树
	preorder(root->left, a, pi);

	//遍历右子树
	preorder(root->right, a, pi);
}

int* preorderTraversal(struct TreeNode* root, int* returnSize)
{
	//*returnSize用于接收二叉树的节点个数
	*returnSize = TreeSize(root);

	//开辟*returnSize个int类型大小的空间用于存放遍历后的结点
	int* a = (int*)malloc(*returnSize * sizeof(int));

	//进行前序遍历
	int i = 0;
	preorder(root, a, &i);

	return a;
}

题六:二叉树遍历

题目描述:

编一个程序,读入用户输入的一串先序遍历字符串,根据此字符串建立一个二叉树(以指针方式存储)。例如如下的先序遍历字符串:ABC##DE#G##F###其中"#"表示的是空格,空格字符代表空树。建立起此二叉树以后,再对二叉树进行中序遍历,输出遍历结果。

分析:

根据所给的先序遍历字符串,采用前序遍历的方式进行二叉树的创建,再将创建好的二叉树进行中序遍历即可。

实现:

//二叉树的定义
typedef char BTDataType;
typedef struct BinaryTreeNode
{
	struct BinaryTreeNode* left;//左子树
	struct BinaryTreeNode* right;//右子树
	BTDataType data;
}BTNode;

//创建结点
BTNode* BuyNode(BTDataType x)
{
	BTNode* node = (BTNode*)malloc(sizeof(BTNode));
	assert(node);

	node->data = x;
	node->left = NULL;
	node->right = NULL;

	return node;
}

//前序创建二叉树
BTNode* CreateTree(char* str, int* pi)
{

	//若结点为空,也就是空树,空树数组的下标也要++,且为它malloc空间进行存储
	if (str[*pi] == '#')
	{
		(*pi)++;
		return NULL;
	}

	//前序构建二叉树
	BTNode* root = BuyNode(str[(*pi)++]);
	root->left = CreateTree(str, pi);
	root->right = CreateTree(str, pi);

	return root;
}

//中序遍历二叉树
void InOrder(BTNode* root)
{
	//判断树是否为空
	if (root == NULL)
	{
		return;
	}

	InOrder(root->left);
	printf("%c ",root->data);
	InOrder(root->right);
}

int main()
{
	char str[100];
	scanf("%s", str);

	int i = 0;
	BTNode* root = CreateTree(str, &i);
	InOrder(root);

	return 0;
}

递归分析:

数据结构初阶--二叉树的链式结构,数据结构,数据结构文章来源地址https://www.toymoban.com/news/detail-650159.html

到了这里,关于数据结构初阶--二叉树的链式结构的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 数据结构——二叉树的链式结构

      个人主页 : 日刷百题 系列专栏 : 〖C语言小游戏〗〖Linux〗〖数据结构〗  〖C语言〗 🌎 欢迎各位 → 点赞 👍+ 收藏 ⭐️+ 留言 📝  ​ 这里我们使用先序遍历的思想来创建二叉树,这里的内容对于刚接触二叉树的朋友可能有些难理解,不妨先看完下面的二叉树各种遍历

    2024年02月05日
    浏览(43)
  • 【数据结构】二叉树的链式结构

    学习链式二叉树要知道三种遍历方式,便于对二叉树的节点以及左子树和右子树进行操作。 前序遍历:根、左子树、右子树 中序遍历:左子树、根、右子树 后序遍历:左子树、右子树、根 以下图为例: 得到的结果: 前序遍历:1 2 3 4 5 6 中序遍历:3 2 1 5 4 6 后序遍历:3 2

    2024年02月08日
    浏览(52)
  • 【数据结构】二叉树的链式存储结构

    前序遍历,又叫先根遍历。 遍历顺序:根 - 左子树 - 右子树 代码: 中序遍历,又叫中根遍历。 遍历顺序:左子树 - 根 - 右子树 代码 : 后序遍历,又叫后根遍历。 遍历顺序:左子树 - 右子树 - 根 代码 : 除了先序遍历、中序遍历、后序遍历外,还可以对二叉树进行层序遍

    2024年02月09日
    浏览(42)
  • 【数据结构 —— 二叉树的链式结构实现】

    树是一种非线性的数据结构,它是由n(n=0)个有限结点组成一个具有层次关系的集合。 把它叫做树是因为它看起来像一棵倒挂的树,也就是说它是根朝上,而叶朝下的。 1.有一个 特殊的结点,称为根结点 ,根节点没有前驱结点 2.除根节点外, 其余结点被分成M(M0)个互不相交

    2024年02月05日
    浏览(54)
  • 【数据结构—二叉树的链式结构实现】

    提示:文章写完后,目录可以自动生成,如何生成可参考右边的帮助文档 文章目录 前言 一、二叉树的存储结构 二、二叉树链式结构的实现 2.1手动构建一课树 2.2二叉树的遍历 三、二叉树链式结构的实现 3.1前序遍历(递归) 3.2中序遍历(递归) 3.3后序遍历(递归) 3.4层序遍历(非递

    2024年02月03日
    浏览(56)
  • 数据结构:链式二叉树初阶

    目录 一.链式二叉树的逻辑结构 1.链式二叉树的结点结构体定义 2.链式二叉树逻辑结构 二.链式二叉树的遍历算法 1.前序遍历 2.中序遍历 3.后序遍历  4.层序遍历(二叉树非递归遍历算法) 层序遍历概念: 层序遍历算法实现思路:  层序遍历代码实现: 三.链式二叉树遍历算法的运用

    2024年02月02日
    浏览(42)
  • 数据结构-二叉树的链式存储

    二叉树的存储结构有顺序结构和链式结构两种,顺序结构我已经在上篇进行了详细的讲解,地址:数据结构-二叉树的顺序存储与堆(堆排序),本篇我们就主要讲解二叉树的链式存储。 二叉树的链式存储结构是指,用链表来表示一棵二叉树,即用链来指示元素的逻辑关系。

    2024年02月02日
    浏览(48)
  • 【数据结构】二叉树的链式结构及实现

    目录 1. 前置说明 2. 二叉树的遍历 2.1 前序、中序以及后序遍历 2.2 层序遍历 3. 节点个数及高度等 4. 二叉树的创建和销毁 在学习二叉树的基本操作前,需先要创建一棵二叉树,然后才能学习其相关的基本操作。由于现在大家对二叉树结构掌握还不够深入,为了降低大家学习成

    2024年02月08日
    浏览(39)
  • 数据结构:二叉树的链式结构的实现

      目录  1.通过前序遍历构建二叉树 2. 二叉树的销毁  3.二叉树的遍历 4. 二叉树的节点个位和二叉树的叶子节点个位数 5. 二叉树的的k层节点数和查找值为x的节点 6. 判断二叉树是否为完全二叉树和求二叉树的高度h 二叉树的前序遍历 二叉树的中序遍历 二叉树的后序遍历

    2024年02月12日
    浏览(43)
  • 【数据结构】二叉树的链式结构的实现 -- 详解

    在学习二叉树的基本操作前,需先要创建一棵二叉树,然后才能学习其相关的基本操作。为了降低大家学习成本,此处手动快速创建一棵简单的二叉树,快速进入二叉树操作学习。 注意 :上述代码并不是创建二叉树的方式。 学习二叉树结构,最简单的方式就是遍历。所谓

    2024年02月12日
    浏览(40)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包