数据结构OJ实验7-树结构及应用

A. 树的先序遍历(双亲转先序)

题目描述

亲表示法结果,用一个二维数组表示,位置下标从0开始,如果双亲位置为-1则表示该结点为根结点

编写程序,输出该树的先根遍历结果。

输入

第一个输入t,表示有t棵树

接着每棵树输入3行:

第1行输入n,表示树有n个结点

第2行输入n个英文字母,表示每个树结点的数值

第3行输入n个整数,表示每个结点的双亲在数组的下标

以此类推输入下一棵树

输出

共输出t行,每行输出一棵树的先根遍历结果

样例查看模式 

正常显示查看格式

输入样例1 

2
7
A B C D E F G
-1 0 0 0 1 1 3
10
A B C D R E F G H K
4 4 4 0 -1 0 2 6 6 6

输出样例1

ABEFCDG
RADEBCFGHK

AC代码

#include<iostream>
#include<vector>
using namespace std;
struct node
{
	char data;
	int fa;
	node()
	{
		data = '0';
		fa = -1;//全部默认自己为根
	}
};
class tree
{
	vector<node>t;
	int n;
public:
	tree()
	{
		cin >> n;
		t.resize(n);
		for (int i = 0; i < n; i++)
		{
			char ch;
			cin >> ch;
			t[i].data = ch;
		}
		for (int i = 0; i < n; i++)
		{
			int x;
			cin >> x;
			if (x != -1)
			{
				t[i].fa = x;
			}
		}
	}
	void pre(int x = -1)
	{
		for (int i = 0; i < n; i++)
		{
			if (t[i].fa == x)
			{
				cout << t[i].data;
				pre(i);
			}
		}
	}
};
int main()
{
	int t;
	cin >> t;
	while (t--)
	{
		tree tt;
		tt.pre();
		cout << endl;
	}
	return 0;
}

B. 树的后根遍历(孩子链表法)

题目描述

根据树的孩子链表表示法构建一棵树,并输出树的后根遍历

下标位置从0开始

输入

第一行输入两个参数,第一个参数n表示树有n个结点,第二个参数r表示根结点的数组下标

接着n行,每行先输入一个结点的数值(用单个字母表示),再输入结点的孩子的下标,最后以-1结尾

如果该结点没有孩子,则一行只输入结点的数值和-1

输出

只有一行输出,树的后根遍历结果

样例查看模式 

正常显示查看格式

输入样例1 

10 4
A 3 5 -1
B -1
C 6 -1
D -1
R 0 1 2 -1
E -1
F 7 8 9 -1
G -1
H -1
K -1

输出样例1

DEABGHKFCR

AC代码

#include<iostream>
#include<vector>
using namespace std;
struct node
{
	char data;
	vector<int>child;//孩子用数组形式表示
};
class tree
{
	vector<node>t;
	int n;
	int root;
	void pos(int x)
	{
		for (int i = 0; i < t[x].child.size(); i++)
		{
			pos(t[x].child[i]);
		}
		cout << t[x].data;//遍历孩子后才输出根
	}
public:
	tree()
	{
		cin >> n >> root;
		t.resize(n);
		for (int i = 0; i < n; i++)
		{
			cin >> t[i].data;
			vector<int>temp;
			int x;
			cin >> x;
			while (x != -1)
			{
				temp.push_back(x);
				cin >> x;
			}
			t[i].child = temp;
		}
	}
	void posorder()
	{
		pos(root);
		cout << endl;
	}
};
int main()
{
	tree tt;
	tt.posorder();
	return 0;
}

C. 树结构转换(先序转双亲)

题目描述

给出一棵二叉树的特定字符先序遍历结果(空子树用字符'#'表示),构建该二叉树,并输出该二叉树的双亲表示法结果

双亲表示法的数组下标从0开始,根结点必定是在下标0元素,且根结点的双亲下标为-1,左右孩子按下标递增顺序排列,

结点下标是层次遍历顺序。

输入

第一个输入t,表示有t棵二叉树

接着t行,每行输入含特定字符的二叉树先序遍历序列

输出

共输出2t行

每棵二叉树输出两行,第一行输出各个结点的数值,第二行输出各结点的双亲下标

样例查看模式 

正常显示查看格式

输入样例1 

3
AB#C##D##
ABD##E##C##
AB##CDW###E#F##
 

输出样例1

A B D C
-1 0 0 1
A B C D E
-1 0 0 1 1
A B C D E W F
-1 0 0 2 2 3 4

AC代码

#include<iostream>
#include<vector>
#include<queue>
using namespace std;
struct node
{
	char data;
	node* l;
	node* r;
	node* fa;
	node(char c, node* ll = NULL, node* rr = NULL, node* ff = NULL)
	{
		data = c;
		l = ll;
		r = rr;
		fa = ff;
	}
};
class tree
{
	node* root;
	vector<node*>a;//存树结点
	vector<int>b;//存父亲下标
	int len;
	void Create(node*& n, node* f)
	{
		char ch;
		cin >> ch;
		if (ch == '#')
		{
			n = NULL;
			return;
		}
		n = new node(ch);
		n->fa = f;
		Create(n->l,n);
		Create(n->r,n);
	}
	int find(node* n)//找父亲下标
	{
		if (n == NULL)
		{
			return -1;//为根节点
		}
		for (int i = 0; i < 1000; i++)//从头找到尾
		{
			if (n == a[i])return i;
		}
	}
public:
	tree()
	{
		root = NULL;
		a.resize(1000);
		b.resize(1000);
	}
	void createtree()
	{
		Create(root, NULL);
	}
	void lerorder()//二叉树转换为树,采用层次遍历
	{
		queue<node*>v;
		int idx = 0;
		if (root != NULL)
		{
			v.push(root);
			while (!v.empty())
			{
				a[idx] = v.front();
				b[idx] = find(v.front()->fa);
				idx++;
				if (v.front()->l != NULL)v.push(v.front()->l);
				if (v.front()->r != NULL)v.push(v.front()->r);
				v.pop();
			}
		}
		len = idx;
	}
	void show()
	{
		for (int i = 0; i < len; i++)
		{
			if (i)cout << " ";
			cout << a[i]->data;
		}
		cout << endl;
		for (int i = 0; i < len; i++)
		{
			if (i)cout << " ";
			cout << b[i];
		}
		cout << endl;
	}
};
int main()
{
	int t;
	cin >> t;
	while (t--)
	{
		tree b;
		b.createtree();
		b.lerorder();
		b.show();
	}
	return 0;
}

D. 树的双亲结构转孩子链表结构

题目描述

给出一棵树的双亲表示法结果,用一个二维数组表示,位置下标从0开始,如果双亲位置为-1则表示该结点为根结点

编写程序,输出该树的孩子链表表示法结果。

输入

输入一棵树的双亲表示法,共3行:

第1行输入n,表示树有n个结点

第2行输入n个英文字母,表示每个树结点的数值

第3行输入n个整数,表示每个结点的双亲在数组的下标

输出

按输入的结点顺序输出n行,每行输出结点孩子链表结果,先输出结点的数值,再输出结点的孩子的下标,以空格隔开,最后一个数据后面也有空格

如果链表为空则输出结点数值后,输出-1带空格,具体看样式

样例查看模式 

正常显示查看格式

输入样例1 

7
A B C D E F G
-1 0 0 0 1 1 3
 

输出样例1

A 1 2 3 
B 4 5 
C -1 
D 6 
E -1 
F -1 
G -1 

AC代码

#include<iostream>
#include<vector>
#include<queue>
using namespace std;
struct node
{
	char data;
	int fa;
	vector<int>child;
	node()
	{
		data = '0';
		fa = -1;
	}
};
class tree
{
	int n;
	vector<node>t;
public:
	tree()
	{
		cin >> n;
		t.resize(n);
		for (int i = 0; i < n; i++)
		{
			char c;
			cin >> c;
			t[i].data = c;
		}
		for (int i = 0; i < n; i++)
		{
			int x;
			cin >> x;
			if (x != -1)
			{
				t[i].fa = x;
				t[x].child.push_back(i);//在这加入下标为x的孩子,因为x表示i的父亲
			}
		}
	}
	void show()
	{
		for (int i = 0; i < n; i++)
		{
			cout << t[i].data << " ";
			if (t[i].child.size() == 0)
			{
				cout << -1 << " " << endl;
			}
			else
			{
				for (int j = 0; j < t[i].child.size(); j++)
				{
					cout << t[i].child[j] << " ";
				}
				cout << endl;
			}
		}
	}
};
int main()
{
	tree b;
	b.show();
	return 0;
}

E. DS森林叶子编码

题目描述

给定一组森林,编写程序生成对应的二叉树,输出这颗二叉树叶结点对应的二进制编码.规定二叉树的左边由0表示,二叉树的右边由1表示。

输入:

N B  表示N个树,每结点最多B个分支

第2行至第N+1行,每个树的先序遍历

输出

每行表示一个叶结点对应的二进制编码.

样例查看模式 

正常显示查看格式

输入样例1 

3 3
A B 0 0 0 C 0 0 0 D 0 0 0
E F 0 0 0 0 0
G H 0 0 0 I J 0 0 0 0 0 0

输出样例1

0 1 1
1 0
1 1 0 1 0

AC代码

#include<bits/stdc++.h>
using namespace std;
struct node
{
    char data;
    node* l;
    node* r;
    node()
    {
        l = NULL;
        r = NULL;
    }
};
node* createtree(string s, int& idx, int b)
{
    node* n = NULL;
    if (idx < s.size() && s[idx] != '0')
    {
        n = new node;
        n->data = s[idx];
        idx += 2;
        int i = 0;
        while (!n->l && i < b)//左孩子为空则继续找 
        {
            n->l = createtree(s, idx, b);
            i++;
        }
        //其他孩子为左孩子的右孩子
        node* temp = n->l;
        for (; i < b; i++)
        {
            if (s[idx] != '0')
            {
                temp->r = createtree(s, idx, b);
                temp = temp->r;
            }
            else idx += 2;
        }
    }
    else idx += 2;
    return n;
}
void buildcode(node* n, string ans)
{
    if (n->l)
        buildcode(n->l, ans + "0");
    if (n->r)
        buildcode(n->r, ans + "1");
    if (!n->l && !n->r)
    {
        cout << ans[0];
        for (int i = 1; i < ans.length(); i++)
            cout << ' ' << ans[i];
        cout << endl;
    }
}
int main()
{
    int n, b;
    cin >> n >> b;
    getchar();
    vector<string>forest(n);
    vector<node*>root(n);//每一串都为一个二叉树
    for (int i = 0; i < n; i++)
    {
        getline(cin, forest[i]);
        int idx = 0;
        root[i] = createtree(forest[i], idx, b);
    }
    for (int i = 0; i < n - 1; i++)
    {
        root[i]->r = root[i + 1];
    }
    buildcode(root[0], "");
    return 0;
}

F. 先序+中序还原二叉树

题目描述

给定一棵二叉树的先序遍历序列和中序遍历序列,要求计算该二叉树的高度。

输入

输入首先给出正整数N(≤50),为树中结点总数。下面两行先后给出先序和中序遍历序列,均是长度为N的不包含重复英文字母(区别大小写)的字符串。

输出

输出为一个整数,即该二叉树的高度。

样例查看模式 

正常显示查看格式

输入样例1 

9
ABDFGHIEC
FDHGIBEAC

输出样例1

5

AC代码

#include<bits/stdc++.h>
using namespace std;
struct node
{
    char data;
    node* l;
    node* r;
    node(char c, node* ll = NULL, node* rr = NULL)
    {
        data = c;
        l = ll;
        r = rr;
    }
};
//已知n个结点,先序序列为p,中序序列m
node* createtree(int n,char p[],char m[])
{
    node* root = NULL;
    if (n == 0)return NULL;
    root = new node(' ');
    root->data = p[0];//由先序找根
    //找到中序序列中的相对应数据
    for (int i = 0; i < n; i++)
    {
        if (p[0] == m[i])//找到中间根
        {
            root->l = createtree(i, p + 1, m);
            root->r = createtree(n - i - 1, p + 1 + i, m + 1 + i);
        }
    }
    return root;
}
int high(node* root)
{
    if (root == NULL)return 0;
    int lefth = high(root->l);
    int righth = high(root->r);
    return max(lefth, righth) + 1;
}
int main()
{
    int n;
    node* root;
    char pre[100], mid[100];
    cin >> n >> pre >> mid;
    root = createtree(n, pre, mid);
    cout << high(root) << endl;
    return 0;
}