二叉树-3

今天解决了人生导师留给我的第一个问题——通过前序遍历和中序遍历序列,求解后序遍历序列。

思路:D&C

对于二叉树,前/后序遍历+中序遍历是可以求解二叉树的结构的。原理很简单,上图(图片均来自学堂在线 “30240184X 数据结构” 课程视频)

微信图片_20170816151256.png

对于这样的一棵树,前序遍历的结构必然是根+左子树+右子树,而中序遍历是左子树+根+右子树。在inorder中查找r(前序遍历的第一),就可以将inorder划分成左子树部分和右子树部分,在preorder中也根据长度找到了划分位置。在postorder中,r肯定在结尾,左右子树部分的位置也可以确定。确定好了各部分的位置,递归到左右子树就可以。

用数组

void preIn_Post(int pre[], int in[], int post[], int len){
    if(len == 1) {
        post[0] = pre[0];
        return;
    }
    int v = pre[0], i;
    for(i = 1; i < len; i ++) if(v == in[i]) break;
    post[len - 1] = v;
    preIn_Post(pre + 1, in, post, i); //左子树
    preIn_Post(pre + 1 + i, in + 1 + i, post + i, len - 1 - i); //右子树
}

只要看清楚下一步递归的三个数组位置和长度就好了。

重建二叉树

不过我还想把二叉树建出来,感觉只有一部之遥了。

void preIn_rebuild(int pre[], int in[], int len, BinNode* pos, int mode){
    if(len == 0) {return;}
    int v = pre[0], i = 0;
    if(len > 1) for(i = 1; i < len; i ++) if(v == in[i]) break;
    if(mode == AS_LC){
        pos = insertAsLc(v, pos);
    }else{
        pos = insertAsRc(v, pos);
    }
    preIn_rebuild(pre + 1, in, i, pos, AS_LC); //左子树
    preIn_rebuild(pre + 1 + i, in + 1 + i, len - 1 - i, pos, AS_RC); //右子树
}

和前面差不多,只不过要在上次生长的位置上添加孩子。(写着写着发现,之前写的insertAsxx都忘了把新插入节点的指针return回来,修了一下。)

但这段代码的问题是,只能在已有一个节点的树上开始。本来我希望将空树也纳入进来,多一个modeAS_ROOT的条件,结果发现,尽管insertAsRoot里面传的是指针的引用,这段里面传的是指针,所以从main拿到的树根指针会被丢掉,新的树根指针我找不到。可是这段代码也并不方便改为传指针的引用。算了,第一步特殊对待吧。

下面还是用那个满二叉树,测试下:

int pre[50] = {0, 1, 3, 7, 8, 4, 9, 10, 2, 5, 11, 12, 6, 13, 14},
    in[50] = {7, 3, 8, 1, 9, 4, 10, 0, 11, 5, 12, 2, 13, 6, 14}, 
    post[50] = {0}, travLen = 15;   
BinNode* re_root = NULL;

int v = pre[0], i = 0;
for(i = 1; i < travLen; i ++) if(v == in[i]) break;
insertAsRoot(v, re_root);
preIn_rebuild(pre + 1, in, i, re_root, AS_LC); //左子树
preIn_rebuild(pre + 1 + i, in + 1 + i, travLen - 1 - i, re_root, AS_RC); //右子树  

travPostR(re_root); printf("\n");    
travLevel(re_root); printf("\n");

用遍历函数走一遍,没问题。

其它

后序遍历+中序遍历生成前序遍历几乎完全相同。
前序 + 后序在一般情况下是不能确定一棵树的。然而在完全二叉树(出度为0或2)的情况下可以,贴图:

微信图片_20170816151304.png

preorder中左子树部分一定是以l节点起始的,可以在postorder中找到l,也就是左右子树的划分位置;同理,postorder中右子树部分一定是以r节点结束的,可以在postorder中找到左右子树的划分位置。递归就可以了。

如果是一般的二叉树,如果这里左或右节点为空,那仅有的一段子树部分,没办法搞清楚是左子树还是右子树。

感谢学堂在线这个课程竟然讲了这么多。要是人生导师知道我是从视频里看来的而不是自己死抠出来的,估计要气得收拾我了。

溜。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 202,905评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,140评论 2 379
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,791评论 0 335
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,483评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,476评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,516评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,905评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,560评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,778评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,557评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,635评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,338评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,925评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,898评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,142评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,818评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,347评论 2 342

推荐阅读更多精彩内容

  • 二叉树的遍历 二叉树的操作有很多种,其中最常用的是二叉树的遍历。二叉树的遍历是指按照某种顺序访问二叉树中的每个结点...
    Richie_ll阅读 420评论 0 4
  • 树的概述 树是一种非常常用的数据结构,树与前面介绍的线性表,栈,队列等线性结构不同,树是一种非线性结构 1.树的定...
    Jack921阅读 4,427评论 1 31
  • 四、树与二叉树 1. 二叉树的顺序存储结构 二叉树的顺序存储就是用数组存储二叉树。二叉树的每个结点在顺序存储中都有...
    MinoyJet阅读 1,493评论 0 7
  • 基于树实现的数据结构,具有两个核心特征: 逻辑结构:数据元素之间具有层次关系; 数据运算:操作方法具有Log级的平...
    yhthu阅读 4,233评论 1 5
  • 1. 链表 链表是最基本的数据结构,面试官也常常用链表来考察面试者的基本能力,而且链表相关的操作相对而言比较简单,...
    Mr希灵阅读 5,607评论 1 17