假设:有一个n个元素的完全二叉树,为了使其成为满二叉树,补全没有孩子的节点是的除了叶节点所有节点都有两个孩子,即最低层皆为-1.

例1:

              1

         2          3

      4     5     -1    6

     -1  -1     -1  -1                -1   -1

补全的节点赋值为-1,表示当前无节点,需转向别的子树。

step 1:首先,对于一棵二叉树,需定义一个节点的类模板:

包括:节点键值、左子树指针、右子树指针

template <typename T>
class binaryTreeNode {
public:
T element;
binaryTreeNode<T>* leftChild;
binaryTreeNode<T>* rightChild; binaryTreeNode(){leftChild = NULL;rightChild = NULL;}
};

step 2:如何创建一棵二叉树呢?根据链表的特性,成员变量为:节点指针类型的mRoot根节点。除此之外,还包括常见的成员函数:如 获取树的当前规模、获取树的深度、打印或输出树及销毁二叉树等操作。

 template <typename T>
class binaryTree {
private:
binaryTreeNode<T>* mRoot;//树根 int getSize(binaryTreeNode<T>*);
int getHeight(binaryTreeNode<T>*); void preOrder(binaryTreeNode<T>*);
void inOrder(binaryTreeNode<T>*);
void postOrder(binaryTreeNode<T>*);
void distroy(binaryTreeNode<T>*&);
binaryTreeNode<T>* AddNode(const T& key,int direction,binaryTreeNode<T>*& root); public:
binaryTree();
virtual ~binaryTree();
binaryTreeNode<T>* Create();//递归的创建二叉树的节点
void AddNode(const T& key,int direction); int getSize();//递归得到树的节点数目
int getHeight();//递归得到树的高度 //递归遍历
void preOrder();//前序遍历
void inOrder();//中序遍历
void postOrder();//后序遍历 //删除二叉树
void distroy(); };

>>首先,怎么在二叉树中插入节点呢?
可以想到的是,插入新节点的时候需要标明 插得是左孩子还是右孩子which,在哪一个父节点下插入where,及插入的键值是多少what,简单的来说,就是WWW问题

 template <typename T>
binaryTreeNode<T>* binaryTree<T>::AddNode(const T& key,int direction,binaryTreeNode<T>*& root)
{
if(direction == )//左孩子
{
if(root->leftChild == NULL){//找到对应的叶节点插入
root->leftChild = new binaryTreeNode<T>(key);
}
else{
root->leftChild = AddNode(key, direction, root->leftChild);
}
} else//右孩子
{
if (root->rightChild == NULL) {//找到相应的叶节点插入
root->rightChild = new binaryTreeNode<T>(key);
}
else{
root->rightChild = AddNode(key, direction, root->rightChild);
}
} return root;
} template <typename T>
void binaryTree<T>::AddNode(const T& key,int direction)
{
AddNode(key, direction, mRoot);
}

了解了这个思路后,可以不用一个个插入节点,用输入流的方式直接创建一棵树,如下程序:

 template <typename T>
binaryTreeNode<T>* binaryTree<T>::Create(){ binaryTreeNode<T>* current = NULL; T val; cin >> val;//输入键值 if(val == -)//标识当前子树为空,转向下一节点
{
return NULL;
} else{//递归的创建左右子树
current = new binaryTreeNode<T>;
current->element = val;
current->leftChild = Create();
current->rightChild = Create();
return current;
}
}

可以发现,-1是一个过渡标识,标明当前从递归左子树 转向 递归右子树。而上述创建程序是一个前序遍历,所谓前序遍历是指:

  1.先访问父节点

  2.递归左子树

  3.递归右子树

时间复杂度是O(N),因为遍历了每一个节点。

>>创建了二叉树后,怎么销毁?其实只要一一删除每个节点即可,考虑到链表结构,我们不能使用下标去删除节点,只能一个个的访问,而二叉树典型的遍历方法有:前序遍历、中序遍历 及 后序遍历。在这里,我们使用后序遍历进行递归删除, 即自下而上的删除。

 /*二叉树的销毁操作:后序遍历删除

 1)不能使用该声明:void distroy(binaryTreeNode<T>* pNode);该声明会创建一个局部的临时对象来保存传递的指针
虽然实参指针和局部指针都执行同一块堆空间,delete局部指针也会删除二叉树结构所占用的堆内存
但是实参指针将出现无所指的状态,出现不可预料的错误
因此传递的是指针的引用,这样才能将实参指针置空。 2)使用递归方法释放节点 */ template <typename T>
void binaryTree<T>::distroy(binaryTreeNode<T>*& pNode)
{
if(pNode)
{
distroy(pNode->leftChild);
distroy(pNode->rightChild);
delete pNode;
pNode = NULL;
}
}
template <typename T>
void binaryTree<T>::distroy()
{
distroy(mRoot);
}

如要删除如上例1中的二叉树,删除过程依次为:4 5 2 6 3 1

>>对于获取树的深度,有一种方法是,获取左右子树的深度,比较子树深度大小,大的那个增1即为树的深度了。当然,也是递归实现。

 template <typename T>
int binaryTree<T>::getHeight()
{
return getHeight(mRoot);
}
/*
获取当前节点的深度
递归的方法首先要设置截止条件,在进行递归操作。
0.约束条件:节点为空
1.递归左子树,每次递归加1
2.递归右子树,每次递归加1
3.比较左右子树深度,更深的子树+1即为当前节点深度。
*/ template <typename T>
int binaryTree<T>::getHeight(binaryTreeNode<T>* node)
{
if(node == NULL)
return ;
else{
int depL = getHeight(node->leftChild);
int depR = getHeight(node->rightChild);
return (depL > depR) ? depL+ : depR+;
} }

>>同理,获取树的规模只要遍历整棵树即可,这里用递归实现。这里仅给出前序遍历,后序遍历和中序遍历类似则不再给出。

 template <typename T>
void binaryTree<T>::preOrder()
{
cout <<"前序遍历: ";
preOrder(mRoot);
cout << endl;
} /*
前序遍历:
1.由于是递归实现,所以要设置截止条件:当前节点为空
2.先访问父节点,再访问左节点,最后访问右孩子 */
template <typename T>
void binaryTree<T>::preOrder(binaryTreeNode<T>* node)
{
if(node == NULL)
return;
else{
cout << node->element <<' ';
preOrder(node->leftChild);
preOrder(node->rightChild);
}
}

对于例1的遍历结果,如下:

输入:1   - -  - -  -  - -
前序遍历:
中序遍历:
后序遍历:
树的高度为: 3
树的节点数目: 6

>>总之呢,创建二叉树的全过程都用到了递归,那么递归到底是什么呢?

从定义上来讲:递归作为一种算法,是让函数/子程序/过程在程序运行过程中调用自身的方法,能够把一个较为复杂的问题经过层层转换,得到一个与原问题相似但是规模大大减小的问题来求解。递归方法大大减少了代码的复杂度。

实现方法:首先递归必须设置一个终止条件,当满足终止条件时,则递归返回。除此之外,则递归调用自身。

二叉树的基本功能实现方法(C++)的更多相关文章

  1. 使用CSS禁止textarea调整大小功能的方法

    这篇文章主要介绍了使用CSS禁止textarea调整大小功能的方法,禁止可以调整textarea大小功能的方法很简单,使用CSS的resize属性即可,需要的朋友可以参考下 如果你使用谷歌浏览器或火狐 ...

  2. WIN8 、WIN7 下IIS7.5、IIS8 的rewrite 伪静态功能设置方法

    原文 WIN8 .WIN7 下IIS7.5.IIS8 的rewrite 伪静态功能设置方法 win7和win8系统都自带有iis的功能.关于IIS的安装,上一篇已经讲述,这里就不重复了. 下面说下在w ...

  3. JAVAEE——BOS物流项目09:业务受理需求分析、创建表、实现自动分单、数据表格编辑功能使用方法和工作单快速录入

    1 学习计划 1.业务受理需求分析 n 业务通知单 n 工单 n 工作单 2.创建业务受理环节的数据表 n 业务通知单 n 工单 n 工作单 3.实现业务受理自动分单 n 在CRM服务端扩展方法根据手 ...

  4. HTML5调用手机摄像机、相册功能 <input>方法

    最近用MUI框架做webapp项目,在有PLUS环境的基础上能直接调用手机底层的API来使用拍照或从相册选择上传功能! 在查资料的时候,想起了另一种用input调用摄像和相册功能的方法,之前没有深入了 ...

  5. jQuery实现购物车计算价格功能的方法

    本文实例讲述了jQuery实现购物车计算价格功能的简易方法,做的比较简单,现分享给大家供大家参考.具体如下: 目的: <%@ page language="java" con ...

  6. Burpsuite神器常用功能使用方法总结

    Burpsuite介绍: 一款可以进行再WEB应用程序的集成攻击测试平台. 常用的功能: 抓包.重放.爆破 1.使用Burp进行抓包 这边抓包,推荐360浏览器7.1版本(原因:方便) 在浏览器设置代 ...

  7. PHPCMS v9的表单向导实现问答咨询功能的方法

    本文主要介绍了在phpcms v9的表单向导里实现问答咨询功能的方法 phpcms v9内容管理系统本身是没有问答模块的,只有表单向导,但表单向导有很大的局限性,通过表单向导,我们只能查看用户提交的信 ...

  8. Ubuntu上开启Apache Rewrite功能的方法

    Ubuntu上开启Apache Rewrite功能的方法 本文介绍ubuntn系统中开启apache的urlrewrite功能的方法. 在Windows上开启Apache的urlRewrite非常简单 ...

  9. JSP使用网站访问人数统计功能,方法与技巧

    实现网站访问人数统计功能的步骤: 创建静态登录页面,并指定表单提交由登录处理页面进行处理. 创建登录处理页面获得登录信息,查询数据库,判断该用户是否注册,如果该用户已注册,把已登录用户的信息保存在一个 ...

随机推荐

  1. mysql参数安全设置

    MySQL安全相关的参数有哪些?该如何配置? 1.MySQL数据安全 innodb_flush_log_at_trx_commit =1 #innodb每次提交事务redo buffer 刷新到red ...

  2. mysql 数据操作 单表查询 group by 聚合函数

    强调: 如果我们用unique的字段作为分组的依据,则每一条记录自成一组,这种分组没有意义 多条记录之间的某个字段值相同,该字段通常用来作为分组的依据 如果按照每个字段都是唯一的进行分组,意味着按照这 ...

  3. centos shell基础 alias 变量单引号 双引号 history 错误重定向 2>&1 jobs 环境变量 .bash_history source配置文件 nohup & 后台运行 cut,sort,wc ,uniq ,tee ,tr ,split, paste cat> 2.txt <<EOF 通配符 glob模式 发邮件命令mail 2015-4-8 第十二节课

    centos shell基础知识 alias  变量单引号 双引号   history 错误重定向 2>&1  jobs  环境变量 .bash_history  source配置文件 ...

  4. thinkphp5手动注册命名空间

    手动注册 命名空间.利用application/config.php配置文件来注册命名空间1:在application目录同级创建一个myExtend文件夹,里面再创建一个myTest文件夹.里面放一 ...

  5. MySQL server has gone away With statement: INSERT INTO `students`......

    mysql出现ERROR : (2006, 'MySQL server has gone away') 的问题意思就是指client和MySQL server之间的链接断开了. 首选分析给出可能出现的 ...

  6. RabittMQ实践(二): RabbitMQ 与spring、springmvc框架集成

    一.RabbitMQ简介 1.1.rabbitMQ的优点(适用范围)1. 基于erlang语言开发具有高可用高并发的优点,适合集群服务器.2. 健壮.稳定.易用.跨平台.支持多种语言.文档齐全.3. ...

  7. xgboost 自定义评价函数(metric)与目标函数

    比赛得分公式如下: 其中,P为Precision , R为 Recall. GBDT训练基于验证集评价,此时会调用评价函数,XGBoost的best_iteration和best_score均是基于评 ...

  8. A Practical Guide to Support Vector Classi cation

    <A Practical Guide to Support Vector Classication>是一篇libSVM使用入门教程以及一些实用技巧. 1. Basic Kernels: ( ...

  9. hdu 5017

    好恶心的题 #include <cstdio> #include <string.h> #include <algorithm> #include <cmat ...

  10. oracle定时器job的使用

    对于DBA来说,数据库Job再熟悉不过了,因为经常要数据库定时的自动执行一些脚本,或做数据库备份,或做数据的提炼,或做数据库的性能优化,包括重建索引等等的工作.但是,Oracle定时器Job时间的处理 ...