完整二叉树实例程序
- 格式:docx
- 大小:62.07 KB
- 文档页数:14
二叉树的建立与基本操作二叉树是一种特殊的树形结构,它由节点(node)组成,每个节点最多有两个子节点。
二叉树的基本操作包括建立二叉树、遍历二叉树、查找二叉树节点、插入和删除节点等。
本文将详细介绍二叉树的建立和基本操作,并给出相应的代码示例。
一、建立二叉树建立二叉树有多种方法,包括使用数组、链表和前序、中序、后序遍历等。
下面以使用链表的方式来建立二叉树为例。
1.定义二叉树节点类首先,定义一个二叉树节点的类,包含节点值、左子节点和右子节点三个属性。
```pythonclass Node:def __init__(self, value):self.value = valueself.left = Noneself.right = None```2.建立二叉树使用递归的方法来建立二叉树,先构造根节点,然后递归地构造左子树和右子树。
```pythondef build_binary_tree(lst):if not lst: # 如果 lst 为空,则返回 Nonereturn Nonemid = len(lst) // 2 # 取 lst 的中间元素作为根节点的值root = Node(lst[mid])root.left = build_binary_tree(lst[:mid]) # 递归构造左子树root.right = build_binary_tree(lst[mid+1:]) # 递归构造右子树return root```下面是建立二叉树的示例代码:```pythonlst = [1, 2, 3, 4, 5, 6, 7]root = build_binary_tree(lst)```二、遍历二叉树遍历二叉树是指按照其中一规则访问二叉树的所有节点,常见的遍历方式有前序遍历、中序遍历和后序遍历。
1.前序遍历前序遍历是指先访问根节点,然后访问左子节点,最后访问右子节点。
```pythondef pre_order_traversal(root):if root:print(root.value) # 先访问根节点pre_order_traversal(root.left) # 递归访问左子树pre_order_traversal(root.right) # 递归访问右子树```2.中序遍历中序遍历是指先访问左子节点,然后访问根节点,最后访问右子节点。
动态规划-最优⼆叉搜索树摘要: 本章介绍了⼆叉查找树的概念及操作。
主要内容包括⼆叉查找树的性质,如何在⼆叉查找树中查找最⼤值、最⼩值和给定的值,如何找出某⼀个元素的前驱和后继,如何在⼆叉查找树中进⾏插⼊和删除操作。
在⼆叉查找树上执⾏这些基本操作的时间与树的⾼度成正⽐,⼀棵随机构造的⼆叉查找树的期望⾼度为O(lgn),从⽽基本动态集合的操作平均时间为θ(lgn)。
1、⼆叉查找树 ⼆叉查找树是按照⼆叉树结构来组织的,因此可以⽤⼆叉链表结构表⽰。
⼆叉查找树中的关键字的存储⽅式满⾜的特征是:设x为⼆叉查找树中的⼀个结点。
如果y是x的左⼦树中的⼀个结点,则key[y]≤key[x]。
如果y是x的右⼦树中的⼀个结点,则key[x]≤key[y]。
根据⼆叉查找树的特征可知,采⽤中根遍历⼀棵⼆叉查找树,可以得到树中关键字有⼩到⼤的序列。
介绍了⼆叉树概念及其遍历。
⼀棵⼆叉树查找及其中根遍历结果如下图所⽰:书中给出了⼀个定理:如果x是⼀棵包含n个结点的⼦树的根,则其中根遍历运⾏时间为θ(n)。
问题:⼆叉查找树性质与最⼩堆之间有什么区别?能否利⽤最⼩堆的性质在O(n)时间内,按序输出含有n个结点的树中的所有关键字?2、查询⼆叉查找树 ⼆叉查找树中最常见的操作是查找树中的某个关键字,除了基本的查询,还⽀持最⼤值、最⼩值、前驱和后继查询操作,书中就每种查询进⾏了详细的讲解。
(1)查找SEARCH 在⼆叉查找树中查找⼀个给定的关键字k的过程与⼆分查找很类似,根据⼆叉查找树在的关键字存放的特征,很容易得出查找过程:⾸先是关键字k与树根的关键字进⾏⽐较,如果k⼤⽐根的关键字⼤,则在根的右⼦树中查找,否则在根的左⼦树中查找,重复此过程,直到找到与遇到空结点为⽌。
例如下图所⽰的查找关键字13的过程:(查找过程每次在左右⼦树中做出选择,减少⼀半的⼯作量)书中给出了查找过程的递归和⾮递归形式的伪代码:1 TREE_SEARCH(x,k)2 if x=NULL or k=key[x]3 then return x4 if(k<key[x])5 then return TREE_SEARCH(left[x],k)6 else7 then return TREE_SEARCH(right[x],k)1 ITERATIVE_TREE_SEARCH(x,k)2 while x!=NULL and k!=key[x]3 do if k<key[x]4 then x=left[x]5 else6 then x=right[x]7 return x(2)查找最⼤关键字和最⼩关键字 根据⼆叉查找树的特征,很容易查找出最⼤和最⼩关键字。
二叉树c语言实现完整代码二叉树是一种非常常见的数据结构,它由节点和边组成,每个节点最多有两个子节点,分别称为左子节点和右子节点。
在计算机科学中,二叉树被广泛应用于搜索、排序、编译器等领域。
本文将介绍如何使用C语言实现二叉树的完整代码。
我们需要定义二叉树的节点结构体。
每个节点包含三个成员变量:数据、左子节点和右子节点。
代码如下:```struct TreeNode {int val;struct TreeNode *left;struct TreeNode *right;};```接下来,我们需要实现二叉树的基本操作,包括创建节点、插入节点、删除节点、查找节点等。
这些操作可以通过递归实现。
代码如下:```// 创建节点struct TreeNode* createNode(int val) {struct TreeNode* node = (struct TreeNode*)malloc(sizeof(struct TreeNode));node->val = val;node->left = NULL;node->right = NULL;return node;}// 插入节点struct TreeNode* insertNode(struct TreeNode* root, int val) {if (root == NULL) {return createNode(val);}if (val < root->val) {root->left = insertNode(root->left, val);} else {root->right = insertNode(root->right, val);}return root;}// 删除节点struct TreeNode* deleteNode(struct TreeNode* root, int val) {if (root == NULL) {return NULL;}if (val < root->val) {root->left = deleteNode(root->left, val);} else if (val > root->val) {root->right = deleteNode(root->right, val);} else {if (root->left == NULL) {struct TreeNode* temp = root->right;free(root);return temp;} else if (root->right == NULL) {struct TreeNode* temp = root->left;free(root);return temp;}struct TreeNode* temp = findMin(root->right); root->val = temp->val;root->right = deleteNode(root->right, temp->val); }return root;}// 查找节点struct TreeNode* searchNode(struct TreeNode* root, int val) {if (root == NULL || root->val == val) {return root;}if (val < root->val) {return searchNode(root->left, val);} else {return searchNode(root->right, val);}}// 查找最小节点struct TreeNode* findMin(struct TreeNode* root) {while (root->left != NULL) {root = root->left;}return root;}```我们需要实现二叉树的遍历操作,包括前序遍历、中序遍历和后序遍历。
c语言将二叉树填充为完全二叉树的方法要将二叉树填充为完全二叉树,需要按照层次遍历方式进行操作。
下面是使用C语言来实现这个过程的方法:#include <stdio.h>#include <stdlib.h>#define MAX_SIZE 100// 二叉树节点结构体typedef struct Node {int data; // 当前节点的数据struct Node* left; // 左子节点指针struct Node* right; // 右子节点指针} Node;// 辅助队列结构体typedef struct Queue {Node* data[MAX_SIZE];int front, rear;} Queue;// 创建新节点Node* createNode(int data) {Node* newNode = (Node*)malloc(sizeof(Node)); newNode->data = data;newNode->left = NULL;newNode->right = NULL;return newNode;}// 插入节点Node* insertNode(Node* root, int data) {if (root == NULL) {return createNode(data);}Queue queue;queue.front = -1;queue.rear = -1;queue.data[++queue.rear] = root;while (queue.front != queue.rear) {Node* temp = queue.data[++queue.front];if (temp->left) {queue.data[++queue.rear] = temp->left; } else {temp->left = createNode(data);return root;}if (temp->right) {queue.data[++queue.rear] = temp->right; } else {temp->right = createNode(data);return root;}}return root;}// 中序遍历打印二叉树void inorderTraversal(Node* root) {if (root == NULL) {return;}inorderTraversal(root->left);printf("%d ", root->data);inorderTraversal(root->right);}int main() {Node* root = createNode(1);// 假设以下数据为二叉树的节点数据int data[] = {2, 3, 4, 5, 6, 7};int dataSize = sizeof(data) / sizeof(data[0]);for (int i = 0; i < dataSize; i++) {root = insertNode(root, data[i]);}printf("填充完全二叉树后的中序遍历结果:");inorderTraversal(root);return 0;}通过以上代码,我们可以将给定的二叉树填充为完全二叉树,并通过中序遍历方式打印出填充完后的结果。
⼆叉树的⼏个经典例题⼆叉树遍历1题⽬描述编⼀个程序,读⼊⽤户输⼊的⼀串先序遍历字符串,根据此字符串建⽴⼀个⼆叉树(以指针⽅式存储)。
例如如下的先序遍历字符串: ABC##DE#G##F### 其中“#”表⽰的是空格,空格字符代表空树。
建⽴起此⼆叉树以后,再对⼆叉树进⾏中序遍历,输出遍历结果。
输⼊描述:输⼊包括1⾏字符串,长度不超过100。
输出描述:可能有多组测试数据,对于每组数据,输出将输⼊字符串建⽴⼆叉树后中序遍历的序列,每个字符后⾯都有⼀个空格。
每个输出结果占⼀⾏。
输⼊abc##de#g##f###输出c b e gd f a思路:递归建树。
每次都获取输⼊字符串的当前元素,根据题⽬要求(先序、中序、后序等+其他限制条件)建树。
再根据题⽬要求输出即可。
1 #include<bits/stdc++.h>2using namespace std;3struct node{4char data;5 node *lchild,*rchild;6 node(char c):data(c),lchild(NULL),rchild(NULL){}7 };8string s;9int loc;10 node* create(){11char c=s[loc++];//获取每⼀个字符串元素12if(c=='#') return NULL;13 node *t=new node(c);//创建新节点14 t->lchild=create();15 t->rchild=create();16return t;17 }18void inorder(node *t){//递归中序19if(t){20 inorder(t->lchild);21 cout<<t->data<<'';22 inorder(t->rchild);23 }24 }25int main(){26while(cin>>s){27 loc=0;28 node *t=create();//先序遍历建树29 inorder(t);//中序遍历并输出30 cout<<endl;31 }32return0;33 }⼆叉树遍历2题⽬描述⼆叉树的前序、中序、后序遍历的定义:前序遍历:对任⼀⼦树,先访问跟,然后遍历其左⼦树,最后遍历其右⼦树;中序遍历:对任⼀⼦树,先遍历其左⼦树,然后访问根,最后遍历其右⼦树;后序遍历:对任⼀⼦树,先遍历其左⼦树,然后遍历其右⼦树,最后访问根。
最早提出遍历问题的是对存储在计算机中的表达式求值。
例如:(a+b ×(c-d))-e/f 。
表达式用树形来表示,如图8-11-1所示。
运算符在树中放在非终端结点的位置上,操作数放在叶子结点处。
当我们对此二叉树进行先序、中序和后序遍历后,便可得到表达式的前缀、中缀和后缀书写形式:前缀:-+a*b-cd/ef中缀:a+b*c-d-e/f 后缀:abcd-*+ef/-其中,中缀形式是算术表达式的通常形式,只是没有括号。
在计算机内,使用后缀表达式易于求值。
例1 输入一个算术表达式,判断该表达式是否合法,若不合法,给出错误信息;若合法,则输出合法表达式的表达式树。
【算法分析】表达式不合法有三种情况:①左右括号不匹配;②变量名不合法;③运算符两旁无参与运算的变量或数。
分析表达式树可以看到:表达式的根结点及其子树的根结点为运算符,其在树中的顺序是按运算的先后顺序从后到前,表达树的叶子为参与运算的变量或数。
表达式树如图8-11-2处理时,首先找到运算级别最低的运算符“+”作为根结点,继而确定该根结点的左、右子树结点在表达式串中的范围为a 和(b-c)/d ,再在对应的范围内寻找运算级别最低的运算符作为子树的根结点,直到范围内无运算符,则剩余的变量或数为表达式树的叶子。
【算法步骤】① 设数组ex 存放表达式串的各字符,lt 、rt 作为结点的左右指针,变量left 、right 用于存放每次取字符范围的左、右界。
② 设置左界初值为1;右界初值为串长度。
③ 判断左右括号是否匹配,不匹配则认为输入有错误。
④ 在表达式的左右界范围内寻找运算级别最低的运算符,同时判断运算符两旁有否参与运算的变量或数。
若无,则输入表达式不合法;若有,作为当前子树的根结点,设置左子树指针及其左右界值,设置右子树指针及其左右界值。
⑤ 若表达式在左右界范围内无运算符,则为叶子结点,判断变量名或数是否合法。
⑥ 转④,直到表达式字符取完为止。
包含二叉树的顺序存储,插入,删除结点,以及设置root结点,和前序、中序、后序、按层次遍历的完成代码。
#include<iostream>#include<stdio.h>#include<stdlib.h>#define QUEUE_MAXSIZE 50using namespace std;typedef struct CTree{char data;struct CTree *left;struct CTree *right;}CBTree;CBTree *BTreeInit(CBTree *node){if(node != NULL)return node;elsereturn NULL;}int BTreeAddNode(CBTree *bt,CBTree *node,int n) {if(bt==NULL){cout<<"no parent!"<<endl;return 0;}switch(n){case 1:if(bt->left){cout<<"the left is existing."<<endl;return 0;}elsebt->left=node;break;case 2:if(bt->right){cout<<"the right is existing."<<endl;return 0;}elsebt->right=node;break;default:cout<<"it's wrong!"<<endl;return 0;}return 1;}CBTree *BTreeLeft(CBTree *bt) {if(bt)return bt->left;elsereturn NULL;}CBTree *BTreeright(CBTree *bt){if(bt)return bt->right;elsereturn NULL;}int BTreeIsempty(CBTree *bt) {if(bt)return 0;elsereturn 1;}int BTreeDepth(CBTree *bt) {int dep1,dep2;if(bt==NULL)return 0;else{dep1=BTreeDepth(bt->left);dep2=BTreeDepth(bt->right);if(dep1>dep2)return dep1+1;elsereturn dep2+1;}}CBTree *BTreeFind(CBTree *bt,char data) {CBTree *p;if(bt==NULL)return NULL;else{if(bt->data == data)return bt;else{if(p=BTreeFind(bt->left,data))return p;else if(p=BTreeFind(bt->right,data))return p;elsereturn NULL;}}}void BTreeClear(CBTree *bt){if(bt){BTreeClear(bt->left);BTreeClear(bt->right);free(bt);bt=NULL;}return;}void BTree_pre(CBTree *bt){{cout<<bt->data<<" ";BTree_pre(bt->left);BTree_pre(bt->right);}return;}void BTree_mid(CBTree *bt) {if(bt){BTree_mid(bt->left);cout<<bt->data<<" ";BTree_mid(bt->right);}return;}void BTree_post(CBTree *bt) {{BTree_post(bt->left);BTree_post(bt->right);cout<<bt->data<<" ";}return;}void BTree_level(CBTree *bt){CBTree *p,*q[QUEUE_MAXSIZE];int head=0,tail=0;if(bt){tail=(tail+1)%QUEUE_MAXSIZE;q[tail]=bt;}while(head != tail){head=(head+1)%QUEUE_MAXSIZE;p=q[head];cout<<p->data<<" ";if(p->left != NULL){tail=(tail+1)%QUEUE_MAXSIZE;q[tail]=p->left;}if(p->right != NULL){tail=(tail+1)%QUEUE_MAXSIZE;q[tail]=p->right;}}return;}CBTree *InitRoot(){CBTree *node;if(node=(CBTree *)malloc(sizeof(CBTree))) {cout<<"input the root: ";cin>>node->data;node->left=NULL;node->right=NULL;return node;}return NULL;}void AddNode(CBTree *bt){CBTree *node,*parent;char data,select;if(node= (CBTree *)malloc(sizeof(CBTree))) {cout<<"please input the data: ";cin>>node->data;node->left=NULL;node->right=NULL;cout<<"please input the parent: ";cin>>data;parent=BTreeFind(bt,data);if(! parent){cout<<"no parent."<<endl;free(node);return;}cout<<"1.to left; 2.to right";do{cin>>select;select=select-'0';if(select==1 || select==2)BTreeAddNode(parent,node,select);} while (select!=1 && select !=2);}return;}主函数调用为:int main(){CBTree *root=NULL;char select;do{cout<<"1.set root 2.add node 3.pre_scan"<<endl;cout<<"4.min_scan 5.post_scan 6.level_scan"<<endl; cout<<"7.the depth 0.exit"<<endl;cin>>select;switch(select){case'1':root=InitRoot();break;case'2':AddNode(root);break;case'3':cout<<"the pre_scan is:";BTree_pre(root);cout<<endl;break;case'4':cout<<"the mid_scan is:";BTree_mid(root);cout<<endl;break;case'5':cout<<"the post_scan is:";BTree_post(root);cout<<endl;break;case'6':cout<<"the level_scan is:";BTree_level(root);cout<<endl;break;case'7':cout<<"the depth is "<<BTreeDepth(root)<<endl;break;case'0':break;}} while (select !='0');BTreeClear(root);root=NULL;return 0;}实例程序:。