数据结构:树的存储结构

学习树之前,我们已经了解了二叉树的顺序存储和链式存储,哪么我们如何来存储普通型的树结构的数据?如下图1:

数据结构:树的存储结构_第1张图片

如图1所示,这是一颗普通的树,我们要如何来存储呢?通常,存储这种树结构的数据的方法有3中:

  1. 双亲表示法。
  2. 孩子表示法。
  3. 孩子兄弟表示法。

双亲表示法

双亲表示法采用顺序表(也就是数组)存储普通树,核心思想:顺序存储各个节点的同时,给各个节点附加一个记录其父节点位置的变量。

注意:根结点没有父节点,因此根结点记录父节点位置的变量一般为-1。

例如,采用双亲表示法存储图1中的普通树,其存储状态如图2所示:

数据结构:树的存储结构_第2张图片

图2存储普通树转化为代码为:

#define MAX_SIZE 100//定义树中结点最大数量
typedef struct node{
    char data;//树中结点的数据
    int parent;//结点的父节点再数组中的位置下标
};

typedef struct {
    node arr[MAX_SIZE];//存放树中所有结点
    int n;//节点数
}Tree;

因此,存储图1中普通树的实现代码为:

#include "iostream"

using namespace std;
#define MAX_SIZE 100//定义树中结点最大数量
typedef struct node{
    char data;//树中结点的数据
    int parent;//结点的父节点再数组中的位置下标
};

typedef struct {
    node arr[MAX_SIZE];//存放树中所有结点
    int n;//节点数
}Tree;

Tree Init(Tree tree){
    cout << "请输入结点个数:" << endl;
    cin >> tree.n;
    cout << "请输入结点的值其双亲位于数组中的位置下标:" << endl;
    for(int i = 0; i < tree.n; i++){
        cin >> tree.arr[i].data >> tree.arr[i].parent;
    }
    return tree;
}

void FindParent(Tree tree){
    char a;
    bool IsFind = false;
    cout << "请输入要查询的节点值:" << endl;
    cin >> a;
    for(int i = 0; i < tree.n; i++){
        if(tree.arr[i].data == a){
            IsFind = true;
            cout << a << "的父节点为" << tree.arr[tree.arr[i].parent].data << ",存储位置下标为" << tree.arr[i].parent << endl;
            break;
        }
    }
}
int main(){
    Tree tree;
    for(int i = 0; i < MAX_SIZE; i++){
        tree.arr[i].parent = -1;
        tree.arr[i].data = ' ';
    }
    tree = Init(tree);
    FindParent(tree);
    return 0;
}

程序运行实例:

请输入结点个数:
10
请输入结点的值其双亲位于数组中的位置下标:
R -1
A 0
B 0
C 0
D 1
E 1
F 3
G 6
H 6
K 6
请输入要查询的节点值:
C
C的父节点为R,存储位置下标为0

孩子表示法

孩子表示法是采用“顺序表+链表”的组合结构,其存储过程是:从树的根结点开始,使用顺序表依次存储树中各个节点,需要注意的是,与双亲表示法不同,孩子表示法会给各个节点配备一个链表,用于存储各个节点的孩子节点位于顺序表中的位置。

如果孩子没有叶子节点,则该节点的链表为空链表。

例如,使用孩子表示法存储图3a中的树,则最终存储状况如图3b所示:

数据结构:树的存储结构_第3张图片

代码实现如下:

#include "iostream"

using namespace std;
#define MAX_SIZE 100//定义树中结点最大数量
typedef struct node{
    int child;//链表中每个结点存储的是数据再数组中存储的位置下标
    struct node *next;
};
typedef struct {
    char data;//结点的数据
    node * FirstChild;//孩子链表的头节点
}box;
typedef struct {
    box arr[MAX_SIZE];//存放树中所有结点
    int n, r;//节点数和树根的位置
}Tree;

Tree Init(Tree tree){
    cout << "请输入结点个数:";
    cin >> tree.n;
    for(int i = 0; i < tree.n; i++){
        cout << "输入第" << i + 1 << "个节点的值:" ;
        cin >> tree.arr[i].data;
        tree.arr[i].FirstChild = new node;
        tree.arr[i].FirstChild->next = NULL;
        cout << "输入结点" << tree.arr[i].data << "的孩子结点的数量:";
        int num = 0;
        cin >> num;
        if(num != 0){
            node *p = tree.arr[i].FirstChild;
            for(int j = 0; j < num; j++){
                node *ptr = new node;
                ptr->next = NULL;
                cout << "输入第" << j + 1 << "个孩子节点在顺序表中的位置: ";
                cin >> ptr->child;
                p->next = ptr;
                p = p->next;
            }
        }
    }
    return tree;
}

void FindKids(Tree tree, char ch){
    bool IsFind = false;
    for(int i = 0; i < tree.n; i++){
        if(tree.arr[i].data == ch){
            node *p = tree.arr[i].FirstChild->next;
            while(p){
                IsFind = true;
                cout << tree.arr[p->child].data;
                p = p->next;
            }
            break;
        }
    }
}
int main(){
    Tree tree;
    tree.r = 0;//默认根结点的下标为0
    for(int i = 0; i < MAX_SIZE; i++){
        tree.arr[i].FirstChild = NULL;
    }
    tree = Init(tree);
    FindKids(tree, 'F');//找出结点F的所有孩子结点
    return 0;
}

程序运行结果如下:

请输入结点个数:10
输入第1个节点的值:R
输入结点R的孩子结点的数量:3
输入第1个孩子节点在顺序表中的位置: 1
输入第2个孩子节点在顺序表中的位置: 2
输入第3个孩子节点在顺序表中的位置: 3
输入第2个节点的值:A
输入结点A的孩子结点的数量:2
输入第1个孩子节点在顺序表中的位置: 4
输入第2个孩子节点在顺序表中的位置: 5
输入第3个节点的值:B
输入结点B的孩子结点的数量:0
输入第4个节点的值:C
输入结点C的孩子结点的数量:1
输入第1个孩子节点在顺序表中的位置: 6
输入第5个节点的值:D
输入结点D的孩子结点的数量:0
输入第6个节点的值:E
输入结点E的孩子结点的数量:0
输入第7个节点的值:F
输入结点F的孩子结点的数量:3
输入第1个孩子节点在顺序表中的位置: 7
输入第2个孩子节点在顺序表中的位置: 8
输入第3个孩子节点在顺序表中的位置: 9
输入第8个节点的值:G
输入结点G的孩子结点的数量:0
输入第9个节点的值:H
输入结点H的孩子结点的数量:0
输入第10个节点的值:K
输入结点K的孩子结点的数量:0
找出节点 F 的所有孩子节点:GHK

使用孩子表示法存储的树结构,正好和双亲表示法相反,适用于 查找某个节点的孩子结点,不适用于查找父节点。

其实,我们也可以将双亲表示法和孩子表示法相互结合,就可以得到如图4所示:

数据结构:树的存储结构_第4张图片

使用图4结果存储普通树,技能快速找到指定节点的父节点,也能快速找到指定结点的孩子结点。

孩子兄弟表示法

孩子兄弟表示法:采用的是链式存储结构,其思想是,从树的根结点开始,依次用链表存储各个节点的孩子结点和兄弟节点。

所以,该链表中的结点包括3个部分,如图5所示:

  1. 结点的值。
  2. 指向孩子结点的指针。
  3. 指向兄弟结点的指针。

在这里插入图片描述

代码表示如下:

typedef struct node{
    char data;
    struct node *ForstChild, *NextSibling;
};

还是以图1为例,使用孩子兄弟表示法进行存储的结果如图所示:

数据结构:树的存储结构_第5张图片

由图我们可以发现,孩子兄弟表示法是用二叉树的左子树存储树的孩子结点,用右子树来存储兄弟结点

孩子兄弟表示法可以作为将普通树转化为二叉树的最有效的方法,同时这个方法也被称之为“二叉树表示法”或者”二叉链表表示法

你可能感兴趣的:(数据结构,数据结构)