文章目录
5.3.1 树的存储结构
5. 左儿子右兄弟链接结构
【数据结构】树与二叉树(十九):树的存储结构——左儿子右兄弟链接结构(树、森林与二叉树的转化)
左儿子右兄弟链接结构通过使用每个节点的三个域(FirstChild、Data、NextBrother)来构建一棵树,同时使得树具有二叉树的性质。具体来说,每个节点包含以下信息:
- FirstChild: 存放指向该节点的大儿子(最左边的子节点)的指针。这个指针使得我们可以迅速找到一个节点的第一个子节点。
- Data: 存放节点的数据。
- NextBrother: 存放指向该节点的大兄弟(同一层中右边的兄弟节点)的指针。这个指针使得我们可以在同一层中迅速找到节点的下一个兄弟节点。
通过这样的结构,整棵树可以用左儿子右兄弟链接结构表示成一棵二叉树。这种表示方式有时候被用于一些特殊的树结构,例如二叉树、二叉树的森林等。这种结构的优点之一是它更紧凑地表示树,而不需要额外的指针来表示兄弟关系。
A
/|
B C D
/
E F
A
|
B -- C -- D
|
E -- F
即:
A
/
B
C
/
E D
F
5.3.2 获取结点的算法
1. 获取大儿子、大兄弟结点
【数据结构】树与二叉树(二十):树获取大儿子、大兄弟结点的算法(GFC、GNB)
2. 搜索给定结点的父亲
【数据结构】树与二叉树(廿四):树搜索给定结点的父亲(算法FindFather)
3. 搜索指定数据域的结点
【数据结构】树与二叉树(廿五):树搜索指定数据域的结点(算法FindTarget)
4. 删除结点及其左右子树
a. 逻辑删除与物理删除
b. 算法DST
c. 算法解析
d. 代码实现
递归释放树
void freeTree(TreeNode* root) {
if (root != NULL) {
freeTree(root->firstChild);
freeTree(root->nextBrother);
free(root);
}
}
算法DS
void DelSubtree(TreeNode* t, TreeNode* p) {
if (t == NULL || p == NULL) {
return;
}
TreeNode* result = NULL;
FindFather(t, p, &result);
if (result == NULL) {
return; // 未找到父亲节点
}
if (result->firstChild == p) {
result->firstChild = p->nextBrother;
freeTree(p);
return;
}
TreeNode* q = result->firstChild;
while (q != NULL && q->nextBrother != p) {
q = q->nextBrother;
}
if (q != NULL) {
q->nextBrother = p->nextBrother;
freeTree(p);
}
}
e. 算法测试
int main() {
// 构建左儿子右兄弟链接结构的树
TreeNode* A = createNode('A');
TreeNode* B = createNode('B');
TreeNode* C = createNode('C');
TreeNode* D = createNode('D');
TreeNode* E = createNode('E');
TreeNode* F = createNode('F');
A->firstChild = B;
B->nextBrother = C;
C->nextBrother = D;
C->firstChild = E;
E->nextBrother = F;
// 要删除的子树的根节点
TreeNode* subtreeRoot = F;
// 使用算法 DelSubtree 删除子树
DelSubtree(A, subtreeRoot);
// 输出删除子树后的树结构
printf("Tree after deleting subtree rooted at %c:n", subtreeRoot->data);
// 层次遍历算法
printf("Level Order: n");
LevelOrder(A);
printf("n");
// 释放树节点
freeTree(A);
return 0;
}
5. 代码整合
#include <stdio.h>
#include <stdlib.h>
// 定义树节点
typedef struct TreeNode {
char data;
struct TreeNode* firstChild;
struct TreeNode* nextBrother;
} TreeNode;
// 创建树节点
TreeNode* createNode(char data) {
TreeNode* newNode = (TreeNode*)malloc(sizeof(TreeNode));
if (newNode != NULL) {
newNode->data = data;
newNode->firstChild = NULL;
newNode->nextBrother = NULL;
}
return newNode;
}
// 释放树节点及其子树
void freeTree(TreeNode* root) {
if (root != NULL) {
freeTree(root->firstChild);
freeTree(root->nextBrother);
free(root);
}
}
// 算法GFC:获取大儿子结点
TreeNode* getFirstChild(TreeNode* p) {
if (p != NULL && p->firstChild != NULL) {
return p->firstChild;
}
return NULL;
}
// 算法GNB:获取下一个兄弟结点
TreeNode* getNextBrother(TreeNode* p) {
if (p != NULL && p->nextBrother != NULL) {
return p->nextBrother;
}
return NULL;
}
// 队列结构
typedef struct QueueNode {
TreeNode* treeNode;
struct QueueNode* next;
} QueueNode;
typedef struct {
QueueNode* front;
QueueNode* rear;
} Queue;
// 初始化队列
void initQueue(Queue* q) {
q->front = NULL;
q->rear = NULL;
}
// 入队列
void enqueue(Queue* q, TreeNode* treeNode) {
QueueNode* newNode = (QueueNode*)malloc(sizeof(QueueNode));
newNode->treeNode = treeNode;
newNode->next = NULL;
if (q->rear == NULL) {
q->front = newNode;
q->rear = newNode;
} else {
q->rear->next = newNode;
q->rear = newNode;
}
}
// 出队列
TreeNode* dequeue(Queue* q) {
if (q->front == NULL) {
return NULL; // 队列为空
}
TreeNode* treeNode = q->front->treeNode;
QueueNode* temp = q->front;
q->front = q->front->next;
free(temp);
if (q->front == NULL) {
q->rear = NULL; // 队列为空
}
return treeNode;
}
// 层次遍历的算法
void LevelOrder(TreeNode* root) {
if (root == NULL) {
return;
}
Queue queue;
initQueue(&queue);
enqueue(&queue, root);
while (queue.front != NULL) {
TreeNode* p = dequeue(&queue);
while (p != NULL) {
// 访问当前结点
printf("%c ", p->data);
// 将大儿子结点入队列
if (getFirstChild(p) != NULL) {
enqueue(&queue, getFirstChild(p));
}
// 移动到下一个兄弟结点
p = getNextBrother(p);
}
}
}
// 算法 FindFather
void FindFather(TreeNode* t, TreeNode* p, TreeNode** result) {
*result = NULL;
if (t == NULL || p == NULL || p == t) {
return;
}
TreeNode* q = t->firstChild;
while (q != NULL) {
if (q == p) {
*result = t;
return;
}
FindFather(q, p, result);
if (*result != NULL) {
return;
}
q = q->nextBrother;
}
}
// 算法 DelSubtree
void DelSubtree(TreeNode* t, TreeNode* p) {
if (t == NULL || p == NULL) {
return;
}
TreeNode* result = NULL;
FindFather(t, p, &result);
if (result == NULL) {
return; // 未找到父亲节点
}
if (result->firstChild == p) {
result->firstChild = p->nextBrother;
freeTree(p);
return;
}
TreeNode* q = result->firstChild;
while (q != NULL && q->nextBrother != p) {
q = q->nextBrother;
}
if (q != NULL) {
q->nextBrother = p->nextBrother;
freeTree(p);
}
}
int main() {
// 构建左儿子右兄弟链接结构的树
TreeNode* A = createNode('A');
TreeNode* B = createNode('B');
TreeNode* C = createNode('C');
TreeNode* D = createNode('D');
TreeNode* E = createNode('E');
TreeNode* F = createNode('F');
A->firstChild = B;
B->nextBrother = C;
C->nextBrother = D;
C->firstChild = E;
E->nextBrother = F;
// 要删除的子树的根节点
TreeNode* subtreeRoot = F;
// 使用算法 DelSubtree 删除子树
DelSubtree(A, subtreeRoot);
// 输出删除子树后的树结构
printf("Tree after deleting subtree rooted at %c:n", subtreeRoot->data);
// 层次遍历算法
printf("Level Order: n");
LevelOrder(A);
printf("n");
// 释放树节点
freeTree(A);
return 0;
}
原文地址:https://blog.csdn.net/m0_63834988/article/details/134618351
本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.7code.cn/show_1961.html
如若内容造成侵权/违法违规/事实不符,请联系代码007邮箱:suwngjj01@126.com进行投诉反馈,一经查实,立即删除!
声明:本站所有文章,如无特殊说明或标注,均为本站原创发布。任何个人或组织,在未征得本站同意时,禁止复制、盗用、采集、发布本站内容到任何网站、书籍等各类媒体平台。如若本站内容侵犯了原著者的合法权益,可联系我们进行处理。