目录
- 题目
- 题目要求
- 示例
- 解答
- 方法一、
- 实现思路
- 时间复杂度和空间复杂度
- 代码
- 方法二、
- 实现思路
- 时间复杂度和空间复杂度
- 代码
题目
二叉树的构建及遍历
题目要求
题目链接
示例
解答
方法一、
先构建二叉树,再中序遍历。
实现思路
按照给出的字符串创建二叉树,先依次访问字符串中的字符,如果遇到不为’#'的字符,就将该结点的值赋值为该字符,然后再创建两个新结点,分别为该结点的左孩子和右孩子,然后递归调用方法,使左孩子和右孩子进行同样操作。如果遇到#字符,就将该结点堆顶值赋值为#,然后直接退出函数,即不再创建该结点的左右孩子结点。
时间复杂度和空间复杂度
代码
#include <stdio.h>
#include <stdlib.h>
typedef char BTDataType;
//定义二叉树的结点
typedef struct BinaryTreeNode
{BTDataType data;struct BinaryTreeNode* left;struct BinaryTreeNode* right;
}BTNode;void CreateBinaryTree(BTNode* root, char** arr)
{//如果**arr为'\0',则说明字符串已经结束if (*(*arr) == '\0'){return;}//如果**arr不为#,就将该结点的值为**arr,同时让(*arr)++//此时arr里面存的还是*arr的地址,但是(*arr)++后,*arr中存的地址已经为字符串中下一个字符的地址。if (*(*arr) != '#'){root->data = *(*arr);(*arr)++;}//如果**arr为#,就将该结点的值为#,然后同样将*arr中存的地址向后移一位。else{root->data=*(*arr);(*arr)++;return;}//如果该结点不为空结点,就创建两个结点,来当作该结点的左右孩子。BTNode* left = (BTNode*)malloc(sizeof(BTNode));BTNode* right = (BTNode*)malloc(sizeof(BTNode));root->left = left;root->right = right;//然后递归使该结点的左右孩子完成上述同样操作。CreateBinaryTree(root->left, arr);CreateBinaryTree(root->right, arr);
}
void InOrder(BTNode* root)
{//如果root结点的值为#,就说明该结点为空结点,不需要打印,直接退出函数if (root->data=='#'){return;}//先遍历该结点的左子树InOrder(root->left);//然后再打印该节点的值printf("%c ", root->data);//然后再遍历该结点的右子树InOrder(root->right);
}
int main() {char arr[100] = { 0 };scanf("%s", arr);//创建二叉树的根节点BTNode* bt = (BTNode*)malloc(sizeof(BTNode));//将字符串中第一个字符的地址赋值给pachar* pa = &arr[0];//将指针pa的地址赋给ppa,即ppa为二级指针,通过*ppa就可以得到pa,即得到字符串中第一个字符的地址。//当*ppa+1时,即*ppa+1指向字符串的第二个字符的地址,所以可以通过*ppa++来访问字符串的每个字符char** ppa = &pa;//将二叉树根节点和ppa当作实参CreateBinaryTree(bt, ppa);InOrder(bt);return 0;
}
方法二、
比第一种方法更简洁易懂
实现思路
和第一种方法相似,都是先按照给出的先序遍历的顺序创建二叉树,然后进行中序遍历。
时间复杂度和空间复杂度
代码
#include <stdio.h>
#include <stdlib.h>
#include <assert.h>
typedef char BTDataType;
//定义二叉树的结点
typedef struct BinaryTreeNode
{BTDataType data;struct BinaryTreeNode* left;struct BinaryTreeNode* right;
}BTNode;//创建一个二叉树结点并返回
BTNode* BuyBTNode(BTDataType x)
{BTNode* newNode = (BTNode*)malloc(sizeof(BTNode));newNode->data=x;newNode->left=NULL;newNode->right=NULL;return newNode;
}BTNode* CreateBinaryTree(char* arr,int* count)
{//如果现在访问的字符为#,则说明该结点为NULL,让(*count)++,即访问下一个字符,然后返回NULLif(arr[*count]=='#'){(*count)++;return NULL;}//如果现在访问的字符不为#,将该字符存入到新创建的结点中,BTNode* root = BuyBTNode(arr[(*count)++]);//然后再将该结点的左孩子和右孩子递归调用该函数//如果该结点有左孩子,则会返回创建的新结点//如果该结点没有左孩子,则会返回NULLroot->left = CreateBinaryTree(arr, count);root->right = CreateBinaryTree(arr, count);//然后将根结点返回return root;
}
void InOrder(BTNode* root)
{if(root==NULL){return ;}InOrder(root->left);printf("%c ",root->data);InOrder(root->right);
}
int main()
{char arr[100]={0};scanf("%s",arr);int count = 0;//CreateBinaryTree函数会返回创建的二叉树的根节点BTNode* bt = CreateBinaryTree(arr, &count);InOrder(bt);
}