用双亲表示法存储的树,如何找最近共同父结点,高分相送

数据出境研究所 2009-08-09 05:00:42
#define MAX_TREE_SIZE 100

typedef struct

{

TElemType data;

int parent; /* 双亲位置域 */

} PTNode;

typedef struct

{

PTNode nodes[MAX_TREE_SIZE];

int n; /* 结点数 */

} PTree;


用双亲表示法存储的树,如何找任意两个节点的最近共同父结点,高手们指点一下,最好有代码,非常感谢
...全文
552 15 打赏 收藏 转发到动态 举报
写回复
用AI写文章
15 条回复
切换为时间正序
请发表友善的回复…
发表回复
urakvv7 2009-08-19
  • 打赏
  • 举报
回复
[Quote=引用 12 楼 bigc2000 的回复:]
写错了

应该是O(m+n)

最后查找 应该是O(min(m,n))
查找的方法即反向同位比较序列,如果不同,则上一个是共同根
[/Quote]

没有必要都走到根。
只记录一个序列,并将其信息存到哈希表中。
在另一个节点向根走的时候同时判断当前节点是否在哈希表中存在。
最坏情况下时间复杂度为m+n。
这实际上是vshuang的思路;-)
linren 2009-08-12
  • 打赏
  • 举报
回复
【程序】
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#define MAX_TREE_SIZE 100

typedef int TElemType;

typedef struct
{
TElemType data;
int parent; /* 双亲位置域 */
}PTNode;

typedef struct
{
PTNode nodes[MAX_TREE_SIZE];
int n; /* 结点数 */
}PTree;


int find(PTree* nds,int node1,int node2){
PTNode a,b;
char* n2path;
int p;

if(node1==node2) return node1;
a=nds->nodes[node1];
b=nds->nodes[node2];
if(a.parent==node2||node2==0) return node2;
if(b.parent==node1||node1==0) return node1;

n2path=(char*)malloc(sizeof(char)*MAX_TREE_SIZE);
memset(n2path,0,sizeof(char)*MAX_TREE_SIZE);
n2path[node2]=1;
n2path[b.parent]=1;
while(b.parent!=0){
b=nds->nodes[b.parent];
n2path[b.parent]=1;
}
p=node1;
while(1){
if(n2path[p]==1){
free(n2path);return p;
}
p=a.parent;
a=nds->nodes[a.parent];
}
}

int main(){
PTree t;
int i,j;
t.n=9;
t.nodes[0].parent=0;
t.nodes[1].parent=0;
t.nodes[2].parent=0;
t.nodes[3].parent=1;
t.nodes[4].parent=1;
t.nodes[5].parent=2;
t.nodes[6].parent=2;
t.nodes[7].parent=3;
t.nodes[8].parent=3;

for(i=0;i<9;i++) for(j=0;j<9;j++) printf("%d,%d=>%d\n",i,j,find(&t,i,j));

return 0;
}

【运行结果】
0,0=>0
0,1=>0
0,2=>0
0,3=>0
0,4=>0
0,5=>0
0,6=>0
0,7=>0
0,8=>0
1,0=>0
1,1=>1
1,2=>0
1,3=>1
1,4=>1
1,5=>0
1,6=>0
1,7=>1
1,8=>1
2,0=>0
2,1=>0
2,2=>2
2,3=>0
2,4=>0
2,5=>2
2,6=>2
2,7=>0
2,8=>0
3,0=>0
3,1=>1
3,2=>0
3,3=>3
3,4=>1
3,5=>0
3,6=>0
3,7=>3
3,8=>3
4,0=>0
4,1=>1
4,2=>0
4,3=>1
4,4=>4
4,5=>0
4,6=>0
4,7=>1
4,8=>1
5,0=>0
5,1=>0
5,2=>2
5,3=>0
5,4=>0
5,5=>5
5,6=>2
5,7=>0
5,8=>0
6,0=>0
6,1=>0
6,2=>2
6,3=>0
6,4=>0
6,5=>2
6,6=>6
6,7=>0
6,8=>0
7,0=>0
7,1=>1
7,2=>0
7,3=>3
7,4=>1
7,5=>0
7,6=>0
7,7=>7
7,8=>3
8,0=>0
8,1=>1
8,2=>0
8,3=>3
8,4=>1
8,5=>0
8,6=>0
8,7=>3
8,8=>8
Press any key to continue

【说明】
8楼的程序有一个问题
忘记判断node1本身了……

这个新的程序是用空间换时间的一种办法
时间复杂度为m+n……
FancyMouse 2009-08-12
  • 打赏
  • 举报
回复
单次询问的话记录层深然后再往上走没问题
询问很多还是tarjan吧
bigc2000 2009-08-11
  • 打赏
  • 举报
回复
写错了

应该是O(m+n)

最后查找 应该是O(min(m,n))
查找的方法即反向同位比较序列,如果不同,则上一个是共同根
bigc2000 2009-08-11
  • 打赏
  • 举报
回复
(1)都走到根,比记录序列设为A,B
(2)对这2个序列选择最短的那个 假设是A
对A中的元素(这里选择反方向,即第一个是根)在B中查找,直到找不到,那么上一个元素就是最近的公共根

复杂度可以看成是O(m+n+mlogn)
qq675927952 2009-08-11
  • 打赏
  • 举报
回复
up LS
linren 2009-08-10
  • 打赏
  • 举报
回复
[Quote=引用 7 楼 vshuang 的回复:]
这种方法思想是正确的。
实现的话。

把 b->rootb 路径上的所有结点用哈希表存储起来。

然后对 a -> roota 上每一个点 进行判断,
如果有一个节点在哈希表中,则说明该点是二者最近共同父节点

[/Quote]

如果用哈希表的话效率会好很多……

8楼的程序的时间复杂度大概是m*n
其中m为node2的深度
n为node1的深度……
linren 2009-08-10
  • 打赏
  • 举报
回复
(上接5楼)

【程序】
#include <stdio.h>
#include <stdlib.h>
#define MAX_TREE_SIZE 100

typedef int TElemType;

typedef struct
{
TElemType data;
int parent; /* 双亲位置域 */
}PTNode;

typedef struct
{
PTNode nodes[MAX_TREE_SIZE];
int n; /* 结点数 */
}PTree;


int find(PTree* nds,int node1,int node2){
PTNode a,b;
int i,k;
int* n2path;

if(node1==node2) return node1;
a=nds->nodes[node1];
b=nds->nodes[node2];
if(a.parent==node2||node2==0) return node2;
if(b.parent==node1||node1==0) return node1;

n2path=(int*)malloc(sizeof(int)*nds->n);
k=0;n2path[k]=node2;
k++;n2path[k]=b.parent;
while(b.parent!=0){
b=nds->nodes[b.parent];
k++;n2path[k]=b.parent;
}
while(1){
for(i=0;i<=k;i++){
if(a.parent==n2path[i]){
free(n2path);return a.parent;
}
}
a=nds->nodes[a.parent];
}
}

int main(){
PTree t;
t.n=9;
t.nodes[0].parent=0;
t.nodes[1].parent=0;
t.nodes[2].parent=0;
t.nodes[3].parent=1;
t.nodes[4].parent=1;
t.nodes[5].parent=2;
t.nodes[6].parent=2;
t.nodes[7].parent=3;
t.nodes[8].parent=3;

printf("%d\n",find(&t,7,4));

return 0;
}

【运行结果】
1
Press any key to continue
  • 打赏
  • 举报
回复
这种方法思想是正确的。
实现的话。

把 b->rootb 路径上的所有结点用哈希表存储起来。

然后对 a -> roota 上每一个点 进行判断,
如果有一个节点在哈希表中,则说明该点是二者最近共同父节点



[Quote=引用 5 楼 linren 的回复:]
可以维护一个结点(b)的路径

每次把当前位置改为上一次的两个结点的父节点位置
然后判断结点(a)是否在结点(b)的路径中

不过有些情况需要仔细的考虑
比如说两个结点为同一结点
一个结点正好是另一个结点的父节点或祖先结点等等(可以把当前结点b也放入路径当中)……
[/Quote]
linren 2009-08-09
  • 打赏
  • 举报
回复
[Quote=引用 5 楼 linren 的回复:]
每次把当前位置改为上一次的两个结点的父节点位置
然后判断结点(a)是否在结点(b)的路径中
[/Quote]
在进行下一次的判断时
把结点b经过的新结点加入到维护的路径当中
linren 2009-08-09
  • 打赏
  • 举报
回复
可以维护一个结点(b)的路径

每次把当前位置改为上一次的两个结点的父节点位置
然后判断结点(a)是否在结点(b)的路径中

不过有些情况需要仔细的考虑
比如说两个结点为同一结点
一个结点正好是另一个结点的父节点或祖先结点等等(可以把当前结点b也放入路径当中)……
oyzdz1988 2009-08-09
  • 打赏
  • 举报
回复
不是这么简单的吧,下面的做法是错误的,下面的只适合两节点在同一层的情况。。。
[Quote=引用 3 楼 abcdef0966 的回复:]
双亲法表示的LCA问题,比较简单吧


C/C++ codePTNode PTLCA(PTNode A, PTNode B, PTree T)
{int Aa_index= A.parent;int Ba_index= B.parent;while ( Aa_index!= Ba_index)
{
PTNode AaNode,BaNode;
AaNode= PTree.nodes[Aa.index];
BaNode= PTree.nodes[Ba.index];
Aa.index= AaNode.parent;
Ba.index= BaNode.parent;
}return PTree.nodes[Aa.index];
}
[/Quote]
abcdef0966 2009-08-09
  • 打赏
  • 举报
回复
双亲法表示的LCA问题,比较简单吧


PTNode PTLCA(PTNode A, PTNode B, PTree T)
{
int Aa_index = A.parent;
int Ba_index = B.parent;
while ( Aa_index != Ba_index)
{
PTNode AaNode,BaNode;
AaNode = PTree.nodes[Aa.index];
BaNode = PTree.nodes[Ba.index];
Aa.index = AaNode.parent;
Ba.index = BaNode.parent;
}
return PTree.nodes[Aa.index];
}
showjim 2009-08-09
  • 打赏
  • 举报
回复
先找到根节点,确定各自的深度,然后:
如果可以用空间换时间,则在找根节点的时间记录各自的双亲节点序列,然后找到第一个同级不同值的节点。
如果节省空间,先找到较深节点与较浅节点同级的节点,然后找到第一个值相同的节点。
acdbxzyw 2009-08-09
  • 打赏
  • 举报
回复
最小公共祖先。
这个。。。还是搜搜吧。
5.1 数的逻辑结构 5.1.1 1、的定义 在中常常将数据元素称为结点 (1)有且仅有一个特定的称为根的结点; (2)当n>1时,除根结点之外的其余结点被分成m(m>0)个互不相交的有限集合T1,T2,•••Tm,其中每个集合又是一棵,并称为这个节点的子。 2、的基本术语: 结点的度、的度 叶子节点、分支结点 孩子节点、分支结点、兄弟节点 路径、路径长度 祖先、子孙 结点的层数、的深度(高度) 层序编号 有序、无序 森林 5.1.2 的抽象数据类型定义 5.1.3的遍历操作 1、前序遍历 的前序遍历操作定义为: 若为空,则空操作返回;否则 (1)访问根结点 (2)按照从左向右的顺序前序遍历根结点的每一棵子。 2、中序遍历 的中序遍历操作定义为: 若为空,则空操作返回;否则 (1)按照从左向右的顺序后序遍历根结点的每一棵子; (2)访问根结点。 3、层序遍历 的层序遍历也称作的广泛遍历,其操作定义为的第一层开始,自上而下逐层遍历,在同一层中,按从左向右的顺序对结点逐个访问。 5.2存储结构 5.2.1 双亲表示法的定义可知,中每个结点都有且仅有一个双亲结点。所以利用这一特性,可以用一维数组来存储各个结点,数组中一个元素对应一个结点,数组元素包括结点的数据信息以及该结点的双亲在数组中的下标。 其中: Data为数据域,存储结点的数据信息; Parent为指针即游标,存储结点的双亲在数组中的小标。 5.2.2孩子表示法 1、多重链表表示法 (1)指针域的个数等于该结点的度。 (2)指针域的个数等于的度。 2、孩子链表表示法 把孩子看成一个线性表,且以单链表存储,称为该结点的孩子链表。则n个结点有n个孩子链表。 孩子节点有两类:孩子节点、表头结点。 5.2.3 双亲孩子表示法 即将双亲表示法和孩子链表表示法相结合的存储方法。仍将各结点的孩子分别组成单链表,同时用一维数组顺序存储中的各结点,数组元素除了包括结点的数据信息和该结点的孩子链表的头指针之外,还增设一个域存储结点的双亲在数组的下标。 5.2.4孩子兄弟表示法 又称二链表表示法,其方法是链表中每个结点除数据域外,还设置了两个指针分别指向该结点的第一个孩子和右兄弟链表的结构: Firstchild data rightsib 指针域,存储第一个孩子结点存储地址 数据域,存储结点的数据信息 指针域,存储结点右兄弟结点存储地址 5.3二叉的逻辑结构 最简单的结构,特别适合计算机处理,而且任何数都可以简单的转换为二叉。(重点内容) 5.3.1二叉的定义 二叉是n(n>=0)个结点的有限集合,该集合或者为空集,或者有一个根节点和两棵互不相交的、分别称为根节点的左子和右子的二叉组成。 二叉具有五种基本形态: 1、空二叉; 2、只有一个根结点; 3、根结点只有左子; 4、根结点只有右子; 5、根结点既有左子又有右子 特殊二叉: 1、斜; 2、满二叉; 3、完全二叉; 5.3.2二叉的基本性质 性质5-1 二叉的第i层上最多有2^(i-1)个结点(i>=1)。 性质5-2 在一棵深度为k的二叉中,最多有2^k-1个结点,最少有k个结点。 性质5-3 在一棵二叉中,如果叶子结点的个数为n0,度为2的结点个数为n2,则n0=n2+1. 性质5-4 具有n个结点的完全二叉的深度为【log2^n】+1。 性质5-5 对一棵具有n个结点的完全二叉中的结点从一开始按层序编号,则对于任意的编号为i(1<=i1,则结点i的双亲的编号为【i/2】;否则结点i是根结点,无双亲。 (2)如果2i<=n,则 结点i的左孩子的编号为2i;否则结点i无左孩子。 (3)如果2i+1<=n,则结点i的右孩子的编号为2i+1,否则结点i无右孩子。 5.3.3 二叉的抽象数据类型定义 同类似,在不同的应用中,二叉的基本操作不尽相同。 5.3.4 二叉的遍历操作 二叉的遍历是指从根节点出发,按照某种次序访问二叉是所有结点,使得每个结点被访问一次且仅被访问一次。由于二叉中每个结点都可能有两个子,因此需要寻一条合适的搜索路径。 1、前序遍历 前序遍历二叉操作定义为: 若为空,则空操作返回;否则 (1)访问根结点 (2)前序遍历根结点的左子 (3)前序遍历根结点的右子 2、中序遍历 中序遍历二叉操作定义为: 若为空,则空操作返回;否则 (1)中序遍历根结点的左子 (2)访问根结点 (3)中序遍历根结点的右子 3、后序遍历 后序遍历根结点的左子 后序遍历根结点的右子 访问根结点 4、层序遍历 二叉的层序遍历是指从二叉的第一层开始,从上之下逐层遍历,在同一层中,按从左到右的顺序对结点逐个访问。 5.4 二叉存储结构及实现 5.4.1 顺序存储结构 具体步骤: (1)将二叉按完全二叉编号。 (2)将二叉中的结点一编号顺序存储到一维数组中。 5.4.2 二叉链表 基本思想: 令二叉的每个结点对应一个链表结点,链表结点除了存放于二叉结点有关的数据信息外,还要设置指示左右孩子的指针。 5.4.3 三叉链表 在二叉链表存储方式下,从某个结点出发可以直接访问它的孩子结点,但要到它的双亲结点,则需要从根节点开始搜索,最坏的情况下,需要遍历整个二叉链表。此时采用三叉链表储存二叉。 其中,data,lchild,rchild三个域的含义同二叉,parent域为指向该结点的双亲结点指针。 5.4.4 线索链表 按照某种遍历次序对二叉进行遍历,可以把二叉中所有结点排成一个线性序列。在集体应用中,有时需要访问二叉中的结点在某种遍历序列中前驱和后继,此时,在存储结构中应该保存结点在某种遍历序列中的前驱和后继信息。 前驱和后继结点的指针称为线索,加上线索的二叉称为线索二叉,加上线索的二叉链表称为线索链表。 5.5 二叉遍历的非递归算法 5.5.1 前序遍历非递归算法 关键:在前序遍历过某个左子后,如何到该结点的右子的根指针。 一般的前序遍历执行过程中,设要遍历二叉的根指针为bt,可能出现两种情况: (1)若bt!=NULL,则表明当前二叉不为空,此时,应输入根结点bt的值并将bt保存到栈中,准备继续遍历bt的左子。 (2)若bt=NULL,则表明以bt为根指针的二叉遍历完毕,并且bt是栈顶指针所指结点的左子,若栈不空,则应根据栈顶指针所指结点到待遍历右子的根指针并赋予bt,以继续遍历下去;若栈空,则表明整个二叉遍历完毕。 5.5.2 中序遍历非递归算法 此算法只是需要将前序遍历的非递归算法中输出的语句cout结点要出入两次栈,出两次栈,这种情况的含义和处理方法为: (1)第一次出栈:只遍历晚左子,右子尚未遍历,则该结点不出栈,利用栈顶结点到它的右子,准备遍历它的右子。 (2)第二次出栈:遍历完右子,该结点出栈,并访问它。 设根指针为bt,则可能有以下两种情况: (1)若bt!=NULL,则bt及标志flag入栈,遍历其左子。 (2)若bt=NULL,此时栈空,则整个遍历结束;若栈不空,则表明栈顶结点的左子或右子已遍历结束。若栈顶点的标志flag=1,则表明栈结点的左子已遍历完毕,将flag修改为2,修改为2,并遍历栈定点的右子;若栈顶结点的标志flag=2,则表明栈结点的右子也遍历完毕,输出栈顶结点。 5.6 、森林与二叉的转换 1.转换为二叉 将一棵转换为二叉的方法为: (1)加线——中所有相邻的兄弟结点之间加一条线; (2)去线——对中的每个节点,只保留它与第一个孩子结点之间的连线,删去它与其他孩子结点之间的连线。 (3)层次调节——以根结点为轴心,将顺时针转动一定角度,使之层次分明。 2.森林转换成二叉 (1)将森林中的每一棵二叉转化成二叉; (2)从第二课二叉开始,依次把后一棵二叉的根结点作为一棵二叉根节点的右孩子,当所有二叉连起来后,此时所得到的二叉就是由森林转换得到的二叉。 3、二叉转换为或森林 (1)加线——若某个结点x是其双亲y的左孩子,则把结点x的右孩子、右孩子的右孩子、……,都与结点y用线连起来; (2)去线——删去原二叉中所有的双亲结点与右孩子结点的连线; (3)层次调整——整理由(1)、(2)两步所得到的或森林,使之层次分明。 (4)森林的遍历 两种遍历方法;前序遍历后续遍历。 5.7 应用举例 5.7.1 二叉的应用举例——哈夫曼及哈夫曼编码 1、哈夫曼也称最优二叉,在实际中有着广泛的应用。 叶子节点的权值 是对叶子结点赋予的一个有意义的数值量。 二叉的带权路径长度 设二叉具有n个带权值的叶子节点,从根节点到叶子节点的路径长度与相应的叶子节点权值的乘积之和叫做二叉的带权路径长度,记为: WPL=EWkLk 哈夫曼 给定一组具有确定权值的叶子结点,可以构造出不同的二叉,将其中带权值路径长度最小的二叉称为哈夫曼。 哈夫曼算法基本思想: (1)初始化:由给定的n个权值构造n棵只有一个根结点的二叉,从而得到一个二叉集合。 (2)选取与合并:在F中选取根结点的权值最小的两棵二叉分别作为左、右子构造一棵新的二叉,这棵新的二叉的根结点的权值为其左右子结点的权值之和。 (3)删除与加入:在F中删除作为左、右子的两棵二叉,并将新建的二叉加入到F中。 (4)重复(2)(3)两步的操作,当集合F只剩下一棵二叉时这棵二叉便是哈夫曼。 2、哈夫曼编码 在进行程序设计时,通常给每一个字符记一个单独的代码来表示一组字符,我们称之为编码。

33,028

社区成员

发帖
与我相关
我的任务
社区描述
数据结构与算法相关内容讨论专区
社区管理员
  • 数据结构与算法社区
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告
暂无公告

试试用AI创作助手写篇文章吧