排序二叉树(建立、查找、删除)

二叉树我们已经非常熟悉了,但是除了寻常的储存数据、遍历结构,我们还能用二叉树做什么呢?

我们都知道不同的遍历方式会对相同的树中产生不同的序列结果,排序二叉树就是利用二叉树的遍历特征实现的特殊树种,也叫二叉查找树。

  1. 排序二叉树从根结点起的每一个结点的左子树元素均小于其自身,右子树元素值均大于其自身
  2. 即任何结点的值均大于其左子树所有元素,均小于其右子树所有元素

如:就是一个排序二叉树,直观的一批,从子树到根结点,永远符合左小右大的规则(中序遍历)

Ⅰ、结构定义

排序二叉树的定义与一般二叉树无异

  1. typedef struct BiTree{
  2. int item;
  3. struct BiTree *lchild,*rchild;
  4. }BiTree;

Ⅱ、排序二叉树的查找

我们先来看一下排序二叉树的查找实现,因为插入在排序二叉树中是实现后续建立、删除结点的基础,因为结点带有顺序,故而遍历条件有所改变,代码如下:

  1. int searchnode(BiTree *t, int key)
  2. {
  3. if(!t)
  4. return ;
  5. if(t->item == key)
  6. {
  7. return ;
  8. }
  9. else
  10. {
  11. if(t->item > key)
  12. return searchnode(t->lchild,key);
  13. if(t->item < key)
  14. return searchnode(t->rchild,key);
  15. }
  16. }

清爽递归,不解释

Ⅲ、二叉排序树的插入

  1. void insertbitree(BiTree **t, int value)
  2. {
  3. if(!searchnode(*t,value))
  4. {
  5. if(*t == NULL)
  6. {
  7. *t = (BiTree *)malloc(sizeof(BiTree));
  8. (*t)->item = value;
  9. (*t)->lchild = NULL;
  10. (*t)->rchild = NULL;
  11. }
  12. else
  13. {
  14. if((*t)->item > value)
  15. insertbitree(&((*t)->lchild), value);
  16. else
  17. insertbitree(&((*t)->rchild), value);
  18. }
  19. }
  20. }

这个插入上来先判断一哈我们现有的树里面有没有这个元素,如果有就不会进入循环,至于插入操作的框架也基本符合中序遍历的操作,只是加上了判断大小

Ⅳ、二叉排序树删除结点(HARD)

轻松愉快的建立、查找排序二叉树的操作完成之后,我们来看看比较困难的删除排序二叉树结点的操作。为什么说它困难呢,相比插入或者查找,删除面对的是一个已经成型的树,我们不仅要考虑怎样去掉这个结点,还要想到按照中序以及数字大小将原有结点按序放到正确位置。

好的,我们先来考虑一下我们可能删除哪几种结点:

第一类:待删除结点只有左子树,没有右子树,可以想见,这种情况下只需要把后续的左子树接到待删除结点的上一个结点上,再释放待删除结点的空间就OK

第二类:带删除结点只有右子树,没有左子树,跟第一类一个道理,这样的操作只需要三行就解决,但是棘手的问题总在短暂的轻松过后

第三类:这一类情况就是大魔王辽,左右孩子一个不缺,手心手背都是肉,哪个也不能少,怎么解决这个问题呢?让我们来看一个例子。

看这个丑不拉叽的排序二叉树,非常体现中序遍历特点

现在我们要删除 34 这个结点,就是我们刚才说的那种第三类情况,左右均有结点,这个时候,我们有这两种方法阔以达成目的

第一种:姑且叫他 牺牲前驱法 ,我们要去掉 34 ,就要把他的前驱拿来顶替这个位置,保持二叉排序树的序,然后当然要检测一下,如果牺牲的这个前驱点(在我们这里是 33 )有子树,还需要把子树和上一级连上(如32),这是第一种方法

  1. 用直接前驱 33 替换 34 
  2. 删除原有的 33 结点
  3. 把结点 32 ,移到原 33 位置

第二种:相信你也猜到了,牺牲后继法,反正兄弟两个要挑一个顶上去,让我们看一哈在这个例子中,怎么个牺牲后继

35 已经被我们放上来辽

  1. 用直接后继 35 替换 34
  2. 删除结点 35 

因为这里的 35 茕茕孑立,没儿没女,所以这个例子的这里不需要连接子树,但是千万注意不要认为所有的替换后继法都不用管子树

好的,方法讲明白了辽,我们代码实现一哈

  1. int Delete(BiTree **t)
  2. {
  3. BiTree *s,*q;
  4. if((*t) -> lchild == NULL)//左子树空的情况
  5. {
  6. q = *t;
  7. *t = (*t)->lchild;
  8. free(q);
  9. }
  10. else if( (*t) -> rchild == NULL)//右子树为空的情况
  11. {
  12. q = *t;
  13. *t = (*t)->rchild;
  14. free(q);
  15. }
  16. else //左右子数均为空
  17. {
  18. q = *t;
  19. s = (*t)->lchild;
  20. while(s->rchild)//循环找到直接前驱
  21. {
  22. q = s;
  23. s = s->rchild;
  24. }
  25. (*t) -> item = s -> item; //结点数据替换
  26. if(q != *t) //接原有左右子树
  27. q->rchild = s->lchild;
  28. else
  29. q->lchild = s->lchild;
  30. free(s);
  31. }
  32. return ;
  33. }
  34. int DeleteBST(BiTree **T, int key)
  35. {
  36.  
  37. if (!*T)
  38. return ;
  39. else
  40. {
  41. if (key == (*T)->item)
  42. return Delete(T);
  43. else if (key < (*T)->item)
  44. return DeleteBST(&(*T)->lchild, key);
  45. else
  46. return DeleteBST(&(*T)->rchild, key);
  47. }
  48. return ;
  49. }

解读见注释

测试用主函数部分:

  1. int main()
  2. {
  3. int i;
  4. BiTree *t = NULL;
  5. int value[] = {,,,,,,,,,};
  6. for(i = ; i < ; i++)
  7. insertbitree(&t,value[i]);
  8. printf("建立序列为:\n");
  9. lar(t);
  10. printf("\n");
  11. printf("删除结点88,结果为:\n");
  12. DeleteBST(&t,);
  13. lar(t);
  14. printf("\n");
  15. return ;
  16. }

呼~完毕

(原创)像极了爱情的详解排序二叉树,一秒get的更多相关文章

  1. Mysql高手系列 - 第8篇:详解排序和分页(order by & limit),及存在的坑

    这是Mysql系列第8篇. 环境:mysql5.7.25,cmd命令中进行演示. 代码中被[]包含的表示可选,|符号分开的表示可选其一. 本章内容 详解排序查询 详解limit limit存在的坑 分 ...

  2. 【原创】视频+文字:详解VBA解决数独问题

    [说在前面]: 之前,我在微信朋友圈看到一个同事发了一个状态,说的是她在家辅导孩子做作业,一个数独的题目,好像没有做出来.我看了下,我也做不出来,后来仔细想了下,花了两个多小时时间,用Python编了 ...

  3. 最全 C 语言常用算法详解-排序-队列-堆栈-链表-递归-树 (面试有用)

    具体 源代码 案例查看github,持续更新中............ github地址:https://github.com/Master-fd/C-Algorithm 1. 二分法查找 2. 冒泡 ...

  4. 【原创】RMQ - ST算法详解

    ST算法: ID数组下标: 1   2   3   4   5   6   7   8   9    ID数组元素: 5   7   3   1   4   8   2   9   8 1.ST算法作 ...

  5. (原创)Python 自动化测试框架详解

    自己折腾了一个python的自动化测试框架,梳理了一下流程,简单分享一下. 项目背景 B/S架构,进行用户界面的自动化测试 工具选择 python开发的自动化测试框架,足够灵活,可以随时根据需求进行变 ...

  6. (原创)白话KMP算法详解

    引子:BF暴力算法 KMP算法知名度相当高,燃鹅其理解难度以及代码实现对于初学数据结构和算法的同学并不友好,经过两天的总结,详细总结KMP算法如下: 初学串的模式匹配时,我们都会接触到,或者说应该能想 ...

  7. 【原创】MongoDB安装配置详解(标注两个坑)

    1.下载安装 3.4正式版([坑]不要最新版,有可能进度卡在这个位置不动,等了半个小时也没什么反映,) http://downloads.mongodb.org/win32/mongodb-win32 ...

  8. 详解如何挑战4秒内百万级数据导入SQL Server(转)

      对于大数据量的导入,是DBA们经常会碰到的问题,在这里我们讨论的是SQL Server环境下百万级数据量的导入,希望对大家有所帮助.51CTO编辑向您推荐<SQL Server入门到精通&g ...

  9. Mysql高手系列 - 第9篇:详解分组查询,mysql分组有大坑!

    这是Mysql系列第9篇. 环境:mysql5.7.25,cmd命令中进行演示. 本篇内容 分组查询语法 聚合函数 单字段分组 多字段分组 分组前筛选数据 分组后筛选数据 where和having的区 ...

随机推荐

  1. 关于A*估价函数的总结

    估价函数的优劣决定一个A*算法的好坏 360百科上是这样说的: (https://baike.so.com/doc/6223470-6436780.html) 关于估价函数h(n)与实际距离d(n)的 ...

  2. [译]GLUT教程 - 安装

    Lighthouse3d.com >> GLUT Tutorial >> Basics >> Setup 你需要什么 要用GLUT库开发程序,你可以下载最新版本3. ...

  3. Spring8中lambda表达式的学习(Function接口、BiFunction接口、Consumer接口)

    代码重构,为了确保功能的等效性,梳理代码时,发现如下代码: public SingleRespTTO fundI(SingleReqTTO request) throws Exception { re ...

  4. html加载顺序以及影响页面二次渲染额的因素

    浏览器请求发往服务器以后,返回HTML页面,页面内容开始渲染,具体的执行顺序为: 1. 浏览器开始载入html代码,发现<head>标签内有一个<link>标签引用外部CSS文 ...

  5. 几个常用的 Git 高级命令

    Git 是一款开源优秀的版本管理工具,它最初由 Linus Torvalds 等人开发,用于管理 Linux Kernel 的版本研发.相关的书籍和教程网上琳琅满目,它们多数都详细的介绍其基本的使用和 ...

  6. Oracle数据库中游标的游标的使用

    本人不喜欢说概念啥的,就直接说明使用方法吧 案例1: DECALRE --声明游标 CURSOR C_USER(C_ID NUMBER) IS SELECT NAME FROM USER WHERE ...

  7. ISCC 2018——write up

    WEB Web1-比较数字大小 直接修改input 标签里的maxlength 为999突破长度限制,使得能输入大于999 的数,然后随便输一个数字就行了 或者post修改值 Web2-Web01 s ...

  8. 【TOJ 5103】Electric Fence(皮克定理)

    描述 In this problem, `lattice points' in the plane are points with integer coordinates. In order to c ...

  9. Linux计算某一列的和

    ll | awk '{print $5}' | egrep -v "^$"| paste -sd+|bc 简单说明: ll:拿到当前目录下所有的文件大小 awk:拿到第几列 egr ...

  10. react脚手架搭建1

    23:01:17 react脚手架搭建 (个人用的是webstorm,所以分享下webstorm中的创建react脚手架项目的方法) 1.创建新项目(前提是下载nodejs环境) 2.下载好在webs ...