【数据结构】二叉树

简介: 【数据结构】二叉树

1.二叉树的遍历

前序,中序,后序遍历

学习二叉树结构,最简单的方式就是遍历。所谓二叉树遍历(Traversal)是按照某种特定的规则,依次对二叉树中的节点进行相应的操作,并且每个节点只操作一次。访问结点所做的操作依赖于具体的应用问题。 遍历是二叉树上最重要的运算之一,也是二叉树上进行其它运算的基础。

  1. 前序遍历——访问根结点的操作发生在遍历其左右子树之前。
  2. 中序遍历——访问根结点的操作发生在遍历其左右子树之中(间)。
  3. 后序遍历——访问根结点的操作发生在遍历其左右子树之后。

1.前序遍历:根,左子树,右子树

2.中序遍历:左子树,根,右子树

3.后序遍历:左子树,右子树,根.

前序遍历访问数据为1,2, 3,NULL(3的左子树),NULL(3的右子树),NULL(2的右子树),4,5,NULL(5的左数),NULL(5的右树),6,NULL(6的左树),NULL(6的右树).

中序遍历访问数据为NULL(3的左子树),3,NULL(3的右子树),2,NULL(2的右子树),1,NULL(5的左子树),5,NULL(5的右子树),4,NULL(6的左子树),6,NULL(6的右子树).

后序遍历访问数据为NULL(3的左子树),NULL(3的右子树),3,NULL(2的右子树),2,NULL(5的左子树),NULL(5的右子树),5,NULL(6的左子树),NULL(6的右子树),6,4,1 .

前序遍历的实现

// 二叉树前序遍历
void PreOrder(BTNode* root)
{
  if (root == NULL)
  {
    printf("NULL  ");
    return;
  }
  printf("%d  ",root->data);
  PreOrder(root->left);
  PreOrder(root->right);
}

前序递归打印数据

二叉树中序遍历

// 二叉树中序遍历
void InOrder(BTNode* root)
{
  if (root == NULL)
  {
    printf("NULL  ");
    return;
  }
  
  PreOrder(root->left);
  printf("%d  ", root->data);
  PreOrder(root->right);
}

二叉树后序遍历

void PostOrder(BTNode* root)
{
  if (root == NULL)
  {
    printf("NULL  ");
    return;
  }
  PostOrder(root->left);
  
  PostOrder(root->right);
  printf("%d  ", root->data);
}

代码实现

include<stdio.h>
#include <stdlib.h>
#include <assert.h>
typedef struct BinaryTreeNode
{
  int data;
  struct BinaryTreeNode* left;
  struct BinaryTreeNode* right;
}BTNode;
BTNode* BuyNode(int x)
{
  BTNode* node = (BTNode*)malloc(sizeof(BTNode));
  if (node == NULL)
  {
    perror("malloc fail");
    return NULL;
  }
  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;
}
// 二叉树前序遍历
void PreOrder(BTNode* root)
{
  if (root == NULL)
  {
    printf("NULL  ");
    return;
  }
  printf("%d  ",root->data);
  PreOrder(root->left);
  PreOrder(root->right);
}
// 二叉树中序遍历
void InOrder(BTNode* root)
{
  if (root == NULL)
  {
    printf("NULL  ");
    return;
  }
  
  InOrder(root->left);
  printf("%d  ", root->data);
  InOrder(root->right);
}
// 二叉树后序遍历
void PostOrder(BTNode* root)
{
  if (root == NULL)
  {
    printf("NULL  ");
    return;
  }
  PostOrder(root->left);
  
  PostOrder(root->right);
  printf("%d  ", root->data);
}
int main()
{
  BTNode* bk=CreatBinaryTree();
  PreOrder(bk);
  printf("\n");
  InOrder(bk);
  printf("\n");
  PostOrder(bk);
  printf("\n");
}

编译运行

二叉树结点的个数

int size = 0;//要统计每次递归的次数也就是二叉树结点的个数,定义 全局变量
// 二叉树节点个数
void BinaryTreeSize(BTNode* root)
{
  if (root == NULL)
  {
    return;
  }
  
    ++size;
  
  BinaryTreeSize(root->left);
  BinaryTreeSize(root->right);
}

求二叉树结点的个数的另一个方法,采用分治的思想。

统计学校的人数,院长问导员,导员问老师,老师问班长,班长统计各班的人数。如果遇到空,返回0,其他情况都返回左子树的人数加右子树的人数加上自己,比如老师说要向导员报告人数,老师必须向导员汇报老师左子树的学生,以及右子树的学生加上自己。

int BinaryTreeSize(BTNode* root)
{
  if (root == NULL)
  {
    return 0;
  }
  
  return BinaryTreeSize(root->left) + BinaryTreeSize(root->right) + 1;
  
}

代码简化

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

二叉树叶结点的个数

叶结点的判断,如果该结点的左节点和右节点都为空的话,这个结点就是叶子结点,并且将该值返回

int BinaryTreeLeafSize(BTNode* root)
{
  if (root == NULL)
  {
    return 0;
  }
  else if (root->left == NULL && root->right == NULL)
  {
    return 1;
  }
  
  return BinaryTreeLeafSize(root->left) + BinaryTreeLeafSize(root->right);
}

二叉树第k层结点个数

二叉树第k层结点个数

子问题:转换成左子树的第k-1层和右子树的第k-1层

结束条件:k==1且结点不为空

// 二叉树第k层节点个数
int BinaryTreeLevelKSize(BTNode* root, int k)
{
  assert(k >= 1);
    if (root == NULL)
    {
      return 0;
    }
    if (k == 1)
    {
      return 1;
    }
    return  BinaryTreeLevelKSize(root->left,k-1)+ BinaryTreeLevelKSize(root->right,k -1);
}

二叉树的深度

返回左右子树中高的+1;

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

方法2:

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

比较方法1和方法2,方法一效率要慢一些,因为方法一比较完左右子树返回的值后(这里算递归了一次)还必须再次递归左子树或者右子树,递归左右子树中大的那一个,时间上会翻倍。

整体代码实现

#include<stdio.h>
#include <stdlib.h>
#include <assert.h>
typedef struct BinaryTreeNode
{
  int data;
  struct BinaryTreeNode* left;
  struct BinaryTreeNode* right;
}BTNode;
BTNode* BuyNode(int x)
{
  BTNode* node = (BTNode*)malloc(sizeof(BTNode));
  if (node == NULL)
  {
    perror("malloc fail");
    return NULL;
  }
  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;
}
// 二叉树前序遍历
void PreOrder(BTNode* root)
{
  if (root == NULL)
  {
    printf("NULL  ");
    return;
  }
  printf("%d  ",root->data);
  PreOrder(root->left);
  PreOrder(root->right);
}
// 二叉树中序遍历
void InOrder(BTNode* root)
{
  if (root == NULL)
  {
    printf("NULL  ");
    return;
  }
  
  InOrder(root->left);
  printf("%d  ", root->data);
  InOrder(root->right);
}
// 二叉树后序遍历
void PostOrder(BTNode* root)
{
  if (root == NULL)
  {
    printf("NULL  ");
    return;
  }
  PostOrder(root->left);
  
  PostOrder(root->right);
  printf("%d  ", root->data);
}
int size = 0;
// 二叉树节点个数
//void BinaryTreeSize(BTNode* root)
//{
//
//  if (root == NULL)
//  {
//    return;
//
//
//  }
//  
//    ++size;
//  
//  BinaryTreeSize(root->left);
//  BinaryTreeSize(root->right);
//
//}
//if (root == NULL)
//{
//  return 0;
//
//
//}
//
//
//
//return BinaryTreeSize(root->left) + BinaryTreeSize(root->right) + 1;
int BinaryTreeSize(BTNode* root)
{
return root == 0 ? NULL : BinaryTreeSize(root->left) + BinaryTreeSize(root->right) + 1;
  
}
// 二叉树叶子节点个数
int BinaryTreeLeafSize(BTNode* root)
{
  if (root == NULL)
  {
    return 0;
  }
  else if (root->left == NULL && root->right == NULL)
  {
    return 1;
  }
  
  return BinaryTreeLeafSize(root->left) + BinaryTreeLeafSize(root->right);
}
// 二叉树第k层节点个数
int BinaryTreeLevelKSize(BTNode* root, int k)
{
  assert(k >= 1);
    if (root == NULL)
    {
      return 0;
    }
    if (k == 1)
    {
      return 1;
    }
    return  BinaryTreeLevelKSize(root->left,k-1)+ BinaryTreeLevelKSize(root->right,k -1);
}
//求二叉树的高度
int BTreeHeight(BTNode* root)
{
  if (root == NULL)
  {
    return 0;
  }
  int left = BTreeHeight(root->left);
  int right = BTreeHeight(root->right);
  return left > right ? left + 1 : right + 1;
}
int main()
{
  BTNode* bk=CreatBinaryTree();
  BinaryTreeSize(bk);
  PreOrder(bk);
  printf("\n");
  InOrder(bk);
  printf("\n");
  PostOrder(bk);
  printf("\n");
  printf("二叉树结点个数%d \n", BinaryTreeSize(bk));
  printf("二叉树叶子结点个数%d \n",BinaryTreeLeafSize(bk));
  printf("二叉树第k层结点数%d ", BinaryTreeLevelKSize(bk,3));
  printf("二叉树深度%d ", BTreeHeight(bk));
}

编译运行

目录
相关文章
|
11天前
|
Java C++
【C++数据结构——树】二叉树的基本运算(头歌实践教学平台习题)【合集】
本关任务:编写一个程序实现二叉树的基本运算。​ 相关知识 创建二叉树 销毁二叉树 查找结点 求二叉树的高度 输出二叉树 //二叉树节点结构体定义 structTreeNode{ intval; TreeNode*left; TreeNode*right; TreeNode(intx):val(x),left(NULL),right(NULL){} }; 创建二叉树 //创建二叉树函数(简单示例,手动构建) TreeNode*create
36 12
|
11天前
|
C++
【C++数据结构——树】二叉树的性质(头歌实践教学平台习题)【合集】
本文档介绍了如何根据二叉树的括号表示串创建二叉树,并计算其结点个数、叶子结点个数、某结点的层次和二叉树的宽度。主要内容包括: 1. **定义二叉树节点结构体**:定义了包含节点值、左子节点指针和右子节点指针的结构体。 2. **实现构建二叉树的函数**:通过解析括号表示串,递归地构建二叉树的各个节点及其子树。 3. **使用示例**:展示了如何调用 `buildTree` 函数构建二叉树并进行简单验证。 4. **计算二叉树属性**: - 计算二叉树节点个数。 - 计算二叉树叶子节点个数。 - 计算某节点的层次。 - 计算二叉树的宽度。 最后,提供了测试说明及通关代
36 10
|
11天前
|
存储 算法 测试技术
【C++数据结构——树】二叉树的遍历算法(头歌教学实验平台习题) 【合集】
本任务旨在实现二叉树的遍历,包括先序、中序、后序和层次遍历。首先介绍了二叉树的基本概念与结构定义,并通过C++代码示例展示了如何定义二叉树节点及构建二叉树。接着详细讲解了四种遍历方法的递归实现逻辑,以及层次遍历中队列的应用。最后提供了测试用例和预期输出,确保代码正确性。通过这些内容,帮助读者理解并掌握二叉树遍历的核心思想与实现技巧。
35 2
|
25天前
|
数据库
数据结构中二叉树,哈希表,顺序表,链表的比较补充
二叉搜索树,哈希表,顺序表,链表的特点的比较
数据结构中二叉树,哈希表,顺序表,链表的比较补充
|
2月前
|
机器学习/深度学习 存储 算法
数据结构实验之二叉树实验基础
本实验旨在掌握二叉树的基本特性和遍历算法,包括先序、中序、后序的递归与非递归遍历方法。通过编程实践,加深对二叉树结构的理解,学习如何计算二叉树的深度、叶子节点数等属性。实验内容涉及创建二叉树、实现各种遍历算法及求解特定节点数量。
112 4
|
2月前
|
C语言
【数据结构】二叉树(c语言)(附源码)
本文介绍了如何使用链式结构实现二叉树的基本功能,包括前序、中序、后序和层序遍历,统计节点个数和树的高度,查找节点,判断是否为完全二叉树,以及销毁二叉树。通过手动创建一棵二叉树,详细讲解了每个功能的实现方法和代码示例,帮助读者深入理解递归和数据结构的应用。
171 8
|
3月前
|
存储 算法 关系型数据库
数据结构与算法学习二一:多路查找树、二叉树与B树、2-3树、B+树、B*树。(本章为了解基本知识即可,不做代码学习)
这篇文章主要介绍了多路查找树的基本概念,包括二叉树的局限性、多叉树的优化、B树及其变体(如2-3树、B+树、B*树)的特点和应用,旨在帮助读者理解这些数据结构在文件系统和数据库系统中的重要性和效率。
40 0
数据结构与算法学习二一:多路查找树、二叉树与B树、2-3树、B+树、B*树。(本章为了解基本知识即可,不做代码学习)
|
3月前
|
存储 算法 搜索推荐
数据结构与算法学习十七:顺序储存二叉树、线索化二叉树
这篇文章主要介绍了顺序存储二叉树和线索化二叉树的概念、特点、实现方式以及应用场景。
51 0
数据结构与算法学习十七:顺序储存二叉树、线索化二叉树
|
3月前
|
Java
【用Java学习数据结构系列】震惊,二叉树原来是要这么学习的(二)
【用Java学习数据结构系列】震惊,二叉树原来是要这么学习的(二)
38 1
|
3月前
|
算法 Java C语言
【用Java学习数据结构系列】震惊,二叉树原来是要这么学习的(一)
【用Java学习数据结构系列】震惊,二叉树原来是要这么学习的(一)
34 1

热门文章

最新文章