算法导论 之 平衡二叉树 - 删除[C语言]
一、前言概述
在之前的博文《算法导论 之 平衡二叉树 - 构造、插入、查询、销毁》和《算法导论 之 平衡二叉树 - 打印》中已经给出了构建、插入、查询、打印和销毁平衡二叉树的C语言实现过程,此篇中出现的相关结构体、宏、枚举、函数可以到以上两篇中查找。之所以现在才来写平衡二叉树的删除操作,主要是其过程相对比较复杂,且测试和实现过程中总是出现各种各样的问题,直到今天才彻底解决,平衡二叉树的处理终于可以告一段落。
二、处理思路
虽然平衡二叉树的节点删除操作的代码比较复杂,代码量也比较大,但是其删除过程只有以下3种情况:[注:只要牢牢把握住以下3点,理解平衡二叉树的删除操作不是太难]① 被删的节点是叶子节点
处理思路:将该节点直接从树中删除,并利用递归的特点和高度的变化,反向推算其父节点和祖先节点是否失衡。如果失衡,则判断是那种类型的失衡(LL、LR、RR、RL),再对失衡节点进行旋转处理,直到根节点或高度不再变化。
② 被删的节点只有左子树或只有右子树
处理思路:将左子树(右子树)替代原有节点的位置,并利用递归的特点和高度的变化,反向推算父节点和祖先节点是否失衡。如果失衡,则判断是那种类型的失衡(LL、LR、RR、RL),再对失衡节点进行旋转处理,直到根节点或高度不再发生变化。
③ 被删的节点既有左子树又有右子树
处理思路:找到被删节点的左子树的最右端的节点rnode,将rnode的值赋给node,再把rnode的左孩子替换rnode的位置,在把rnode释放掉,并利用递归特点,反向推断rnode的父节点和祖父节点是否失衡。如果失衡,则判断是哪种类型的失衡(LL、LR、RR、RL),再对失衡的节点进行旋转处理,直到根节点或高度不再发生变化。
三、代码实现
/****************************************************************************** **函数名称: avl_delete **功 能: 删除key值节点(对外接口) **输入参数: ** tree: 平衡二叉树 ** key: 被删除的关键字 **输出参数: NONE **返 回: AVL_SUCCESS:成功 AVL_FAILED:失败 **实现描述: **注意事项: **作 者: # Qifeng.zou # 2013.12.19 # ******************************************************************************/int avl_delete(avl_tree_t *tree, int key){ bool lower = false; /* 记录高度是否降低 */ if(NULL == tree->root) { return AVL_SUCCESS; } return avl_search_and_delete(tree, tree->root, key, &lower);}代码1 删除节点(外部接口)
/****************************************************************************** **函数名称: avl_search_and_delete **功 能: 搜索并删除指定的key值节点(内部接口) **输入参数: ** tree: 平衡二叉树 ** node: 以node为根节点的子树 ** key: 被删除的关键字 **输出参数: ** lower: 高度是否降低 **返 回: AVL_SUCCESS:成功 AVL_FAILED:失败 **实现描述: **注意事项: **作 者: # Qifeng.zou # 2013.12.19 # ******************************************************************************/int avl_search_and_delete(avl_tree_t *tree, avl_node_t *node, int key, bool *lower){ avl_node_t *parent = node->parent; /* 1. 查找需要被删除的节点 */ if(key < node->key) /* 左子树上查找 */ { if(NULL == node->lchild) { return AVL_SUCCESS; } avl_search_and_delete(tree, node->lchild, key, lower); if(true == *lower) { return avl_delete_left_balance(tree, node, lower); } return AVL_SUCCESS; } else if(key > node->key) /* 右子树上查找 */ { if(NULL == node->rchild) { return AVL_SUCCESS; } avl_search_and_delete(tree, node->rchild, key, lower); if(true == *lower) { return avl_delete_right_balance(tree, node, lower); } return AVL_SUCCESS; } /* 2. 已找到将被删除的节点node */ /* 2.1 右子树为空, 只需接它的左子树(叶子节点也走这) */ if(NULL == node->rchild) { *lower = true; avl_replace_child(tree, parent, node, node->lchild); free(node), node = NULL; return AVL_SUCCESS; } /* 2.2 左子树空, 只需接它的右子树 */ else if(NULL == node->lchild) { *lower = true; avl_replace_child(tree, parent, node, node->rchild) free(node), node=NULL; return AVL_SUCCESS; } /* 2.3 左右子树均不为空: 查找左子树最右边的节点 替换被删的节点 */ avl_replace_and_delete(tree, node, node->lchild, lower); if(true == *lower) { avl_print(tree); return avl_delete_left_balance(tree, node, lower); } return AVL_SUCCESS;}代码2 查找并删除节点(内部接口)
/****************************************************************************** **函数名称: avl_replace_and_delete **功 能: 找到替换节点, 并替换被删除的节点(内部接口) **输入参数: ** tree: 平衡二叉树 ** dnode: 将被删除的节点 ** rnode: 此节点最右端的节点将会用来替换被删除的节点. **输出参数: ** lower: 高度是否变化 **返 回: AVL_SUCCESS:成功 AVL_FAILED:失败 **实现描述: **注意事项: ** >> 在此其实并不会删除dnode, 而是将rnode的值给了dnode, 再删了rnode. ** 因为在此使用的递归算法, 如果真把dnode给释放了,会造成压栈的信息出现错误! **作 者: # Qifeng.zou # 2013.12.19 # ******************************************************************************/int avl_replace_and_delete(avl_tree_t *tree, avl_node_t *dnode, avl_node_t *rnode, bool *lower){ avl_node_t *parent = NULL, *rparent = NULL; if(NULL == rnode->rchild) { *lower = true; parent = dnode->parent; rparent = rnode->parent; dnode->key = rnode->key; /* 注: 将rnode的值给了dnode */ if(rnode == dnode->lchild) { avl_set_lchild(dnode, rnode->lchild); /* rnode->parent == dnode节点可能失衡,此处理交给前栈的函数处理 */ } else { avl_set_rchild(rparent, rnode->lchild); /* rnode的父节点可能失衡,此处理交给前栈的函数处理 */ } free(rnode), rnode=NULL; /* 注意: 释放的不是dnode, 而是rnode */ return AVL_SUCCESS; } avl_replace_and_delete(tree, dnode, rnode->rchild, lower); if(true == *lower) { /* dnode的父节点可能失衡,此处理交给前栈的函数处理 但dnode可能使用,因此必须在此自己处理 */ avl_delete_right_balance(tree, rnode, lower); } return AVL_SUCCESS;}代码3 替换并删除节点(内部接口)
/****************************************************************************** **函数名称: avl_delete_left_balance **功 能: 节点node的左子树某节点被删除, 左高度降低后, 平衡化处理(内部接口) **输入参数: ** tree: 平衡二叉树 ** node: 节点node的左子树的某个节点已被删除 **输出参数: ** lower: 高度是否变化 **返 回: AVL_SUCCESS:成功 AVL_FAILED:失败 **实现描述: **注意事项: **作 者: # Qifeng.zou # 2013.12.19 # ******************************************************************************/int avl_delete_left_balance(avl_tree_t *tree, avl_node_t *node, bool *lower){ avl_node_t *rchild = NULL, *rlchild = NULL, *parent = node->parent; switch(node->bf) { case LH: /* 左高: 左子树高度减1 树变矮 */ { node->bf = EH; *lower = true; break; } case EH: /* 等高: 左子树高度减1 树高度不变 */ { node->bf = RH; *lower = false; break; } case RH: /* 右高: 左子树高度减1 树失去平衡 */ { rchild = node->rchild; switch(rchild->bf) { case EH: /* RR型: 向左旋转 */ case RH: /* RR型: 向左旋转 */ { if(EH == rchild->bf) { *lower = false; rchild->bf = LH; node->bf = RH; } else if(RH == rchild->bf) { *lower = true; rchild->bf = EH; node->bf = EH; } avl_set_rchild(node, rchild->lchild); avl_set_lchild(rchild, node); avl_replace_child(tree, parent, node, rchild); break; } case LH: /* RL型: 先向右旋转 再向左旋转 */ { *lower = true; rlchild = rchild->lchild; switch(rlchild->bf) { case LH: { node->bf = EH; rchild->bf = RH; rlchild->bf = EH; break; } case EH: { node->bf = EH; rchild->bf = EH; rlchild->bf = EH; break; } case RH: { node->bf = LH; rchild->bf = EH; rlchild->bf = EH; break; } } avl_set_rchild(node, rlchild->lchild); avl_set_lchild(rchild, rlchild->rchild); avl_set_lchild(rlchild, node); avl_set_rchild(rlchild, rchild); avl_replace_child(tree, parent, node, rlchild); break; } } break; } } return AVL_SUCCESS;}代码4 左子树高度降低后平衡化处理
/****************************************************************************** **函数名称: avl_delete_right_balance **功 能: 节点node的右子树某节点被删除, 左高度降低后, 平衡化处理(内部接口) **输入参数: ** tree: 平衡二叉树 ** node: 节点node的右子树的某个节点已被删除 **输出参数: ** lower: 高度是否变化 **返 回: AVL_SUCCESS:成功 AVL_FAILED:失败 **实现描述: **注意事项: **作 者: # Qifeng.zou # 2013.12.19 # ******************************************************************************/int avl_delete_right_balance(avl_tree_t *tree, avl_node_t *node, bool *lower){ avl_node_t *lchild = NULL, *lrchild = NULL, *parent = node->parent; switch(node->bf) { case LH: /* 左高: 右子树高度减1 树失去平衡 */ { lchild = node->lchild; switch(lchild->bf) { case EH: /* LL型: 向右旋转 */ case LH: /* LL型: 向右旋转 */ { if(EH == lchild->bf) { *lower = false; lchild->bf = RH; node->bf = LH; } else { *lower = true; lchild->bf = EH; node->bf = EH; } avl_set_lchild(node, lchild->rchild); avl_set_rchild(lchild, node); avl_replace_child(tree, parent, node, lchild); break; } case RH: /* LR型: 先向左旋转 再向右旋转 */ { *lower = true; lrchild = lchild->rchild; switch(lrchild->bf) { case LH: { node->bf = RH; lchild->bf = EH; lrchild->bf = EH; break; } case EH: { node->bf = EH; lchild->bf = EH; lrchild->bf = EH; break; } case RH: { node->bf = EH; lchild->bf = LH; lrchild->bf = EH; break; } } avl_set_lchild(node, lrchild->rchild); avl_set_rchild(lchild, lrchild->lchild); avl_set_rchild(lrchild, node); avl_set_lchild(lrchild, lchild); avl_replace_child(tree, parent, node, lrchild); break; } } break; } case EH: /* 等高: 右子树高度减1 树高度不变 */ { node->bf = LH; *lower = false; break; } case RH: /* 右高: 右子树高度减1 树变矮 */ { node->bf = EH; *lower = true; break; } } return AVL_SUCCESS;}代码5 右子树高度降低后 平衡化处理
/* # 检测节点的指针是否存在异常 # 很有效! */void avl_assert(avl_node_t *node){ if((NULL == node) || (NULL == node->parent)) { return; } if((node->parent->lchild != node) && (node->parent->rchild != node)) { assert(0); } if((node->parent == node->lchild) || (node->parent == node->rchild)) { assert(0); }}代码6 节点检测
四、处理结果
左图为原始平衡二叉树,随机删除多个节点后,得到右图。通过观察可知右图依然是一棵平衡二叉树。图1 测试结果
—— 邹祁峰
2013年12月20日 12时
>更多相关文章
- 09-29如何通过wrap malloc定位C/C++程序的内存泄漏
- 02-25打车软件大战升级,补贴还能维持多久?
- 12-23BMP文件右旋90度[c语言]
- 12-23寻找直方图中面积最大的矩形(C语言版)
- 12-23[ndk,2]ndk开发案例和错误处理
- 12-23[ndk,1]ndk开发,C语言入门讲解
- 12-23C语言连续存储实现队列机制
- 12-23Objective-c 数据类型
首页推荐
佛山市东联科技有限公司一直秉承“一切以用户价值为依归
- 01-11全球最受赞誉公司揭晓:苹果连续九年第一
- 12-09罗伯特·莫里斯:让黑客真正变黑
- 12-09谁闯入了中国网络?揭秘美国绝密黑客小组TA
- 12-09警示:iOS6 惊现“闪退”BUG
- 09-19国家邮政局:中秋假期全国共揽投快递包裹超
- 09-19报告:2023年中国数字音乐市场总规模约为19
- 09-19近30亿人次出行,全国铁路客运量创新高
- 09-19阿里CEO吴泳铭:AI计算正在加速演进,成为计
- 08-29市场持续扩大,7月网约车订单信息破10亿单
相关文章
24小时热门资讯
24小时回复排行
热门推荐
最新资讯
操作系统
黑客防御