征战二叉树-第一站

由于基础比较差,什么设计模式,数据结构,操作系统。。。均没有学过,着实让人着急。没办法,只能耐着性子慢慢来。把遇到的东西一点一点搞懂。本篇就来一点提升内功的,数据结构中的二叉树,注意,不是二叉查找树,后面我还会练习二叉查找树。

1. 二叉树?

看一下wiki上的说明

In computer science, a binary tree is a tree data structure in which each node has at most two children, which are referred to as the left child and the right child.

意思是说二叉树是一种数据结构,每个节点最多有两个子节点,也就是左右两个子节点。当然,也有三叉树,四叉树。。。n叉树,只是二叉树我们最常用。


特点:
1、每个结点最多有两颗子树,结点的度最大为2,所以树的度也是2。
2、左子树和右子树是有顺序的,次序不能颠倒。
3、即使某结点只有一个子树,也要区分左右子树。

种类:
1、满二叉树
2、完全二叉树
3、非完全二叉树

这里不给出具体的定义:满二叉树很好理解,完全二叉树主要要学会跟满二叉树对比,就可以判断出是否是完全二叉树,除了完全二叉树之外的数,都是非完全二叉树。

搜了下,二叉树的各种性质,看的有点头大。我就以简单的用法入手,看代码中怎么实现简单的定义与使用,下面就来看代码。
对于二叉树的定义相对比较简单,设置左右节点即可。

public class TreeNode<T>{

    private T data;
    private TreeNode left;
    private TreeNode right;

}

2. 二叉树基本算法

(1) 查找最大深度

/**
     * 获取最大深度
     * @param root
     * @return
     */
    public int getMaxDepth(TreeNode<T> root){

        if (root == null) {
            return 0;
        }
        int left = getMaxDepth(root.left);
        int right = getMaxDepth(root.right);

        return 1 + Math.max(left, right);
    }

(2) 查找最小深度

/**
     * 获取最小深度
     * @param root
     * @return
     */
    public int getMinDepth(TreeNode<T> root) {

        if (root == null) {
            return 0;
        }

        if (root.left == null && root.right == null) {
            return 1;
        }
        return Math.min(getMinDepth(root.left), getMinDepth(root.right)) + 1;
    }

(3) 获取二叉树节点总数

/**
     * 获取节点总数
     * @param root
     * @return
     */
    public int totalNodes(TreeNode<T> root){
        if (root == null) {
            return 0;
        }
        int left = totalNodes(root.left);
        int right = totalNodes(root.right);

        return left + right + 1;
    }

(4) 获取叶子节点总数

/**
     * 获取所有叶子节点
     * @param root
     * @return
     */
    public int totalChildNodes(TreeNode<T> root){

        if (root == null) {
            return 0;
        }
        if (root.left == null && root.right == null) {
            return 1;
        }
        return totalChildNodes(root.left) + totalChildNodes(root.right);
    }

(5) 获取K层节点数

/**
     * 获取K层节点数
     * @param root
     * @param k
     * @return
     */
    public int getLevelKNodes(TreeNode<T> root,int k){

        if (root == null || k < 1) {
            return 0;
        }
        if (k == 1) {
            return 1;
        }
        int left = getLevelKNodes(root.left, k-1);
        int right = getLevelKNodes(root.right, k-1);
        return left + right;
    }

(6) 层次遍历,每一层从左向右遍历

要点:

1.利用队列添加每一层节点,进行遍历

2.如果当前节点存在子节点,加到队列尾部,下次循环遍历取出

/**
     * 层次遍历,每一层从左向右遍历
     * @param root
     * @return
     */
    public List<List<T>> printTreeNodes(TreeNode<T> root) {

        List<List<T>> result = new ArrayList<List<T>>();
        if (root == null) {
            return result;
        }
        LinkedList<TreeNode<T>> linkedList = new LinkedList<TreeNode<T>>();
        linkedList.add(root);

        while (!linkedList.isEmpty()) {

            //用于装每一层的节点值
            ArrayList<T> row = new ArrayList<T>();

            //遍历每一层
            int size = linkedList.size();
            for (int i = 0; i < size; i++) {
                TreeNode<T> node = linkedList.removeFirst();
                row.add(node.getData());

                if (node.left != null) {
                    linkedList.add(node.left);
                }
                if (node.right != null) {
                    linkedList.add(node.right);
                }
            }
            result.add(row);
        }

        return result;
    }

(7) 计算二叉树最大宽度

要点:

过程同层次遍历类似,遍历每一层时,比较每一层的最大宽度,作为二叉树的最大宽度

/**
     * 计算二叉树的最大宽度,通过使用队列计算,每一层的节点数即为每一层的宽度取出最大的一层的宽度
     * @param root
     * @return
     */
    public int getBinaryTreeWidth(TreeNode<T> root){
        if (root == null) {
            return 0;
        }
        LinkedList<TreeNode<T>> queue = new LinkedList<TreeNode<T>>();
        queue.add(root);
        int width = 1;

        while (!queue.isEmpty()) {
            int size = queue.size();
            if (width < size) {
                width = size;
            }
            for (int i = 0; i < size; i++) {

                TreeNode<T> currentNode = queue.removeFirst();
                if (currentNode.left != null) {
                    queue.add(currentNode.left);
                }
                if (currentNode.right != null) {
                    queue.add(currentNode.right);
                }
            }
        }
        return width;
}

(8) 判断两个二叉树是否相同

/**
     * 判断两个二叉树是否相同
     * @param root1
     * @param root2
     * @return
     */
    @SuppressWarnings("null")
    public boolean isSameTreeNode(TreeNode<T> root1,TreeNode<T> root2){

        if (root1 == null && root2 == null) {
            return true;
        }
        else if(root1 != null || root2 != null){
            return false;
        }
        else if (root1.data != root2.data) {
            return false;
        }

        boolean left = isSameTreeNode(root1.left, root2.left);
        boolean right = isSameTreeNode(root1.right, root2.right);
        return left && right;
    }

(9) 前序遍历(递归)

/**
     * 前序遍历
     * @param root
     * @return
     */
    public List<T> preOrder(TreeNode<T> root){

        List<T> list = new ArrayList<T>();
        if (root == null) {
            return list;
        }
        preOder(list, root);
        return list;
    }

    private void preOder(List<T> list,TreeNode<T> root){

        if (root == null) {
            return;
        }
        list.add(root.getData());
        preOder(list, root.left);
        preOder(list, root.right);

    }

(10) 后序遍历(递归)

/**
 * 后序遍历
 * @param root
 * @return
 */
public List<T> postOrder(TreeNode<T> root){

  List<T> list = new ArrayList<T>();
  if (root == null) {
    return list;
  }
  postOder(list, root);
  return list;
}

private void postOder(List<T> list,TreeNode<T> root){

  if (root == null) {
    return;
  }

  postOder(list, root.left);
  postOder(list, root.right);
  list.add(root.getData());

}

(11) 中序遍历(递归)

/**
 * 中序遍历
 * @param root
 * @return
 */
public List<T> InOrder(TreeNode<T> root){

  List<T> list = new ArrayList<T>();
  if (root == null) {
    return list;
  }
  InOder(list, root);
  return list;
}

private void InOder(List<T> list,TreeNode<T> root){

  if (root == null) {
    return;
  }

  InOder(list, root.left);
  list.add(root.getData());
  InOder(list, root.right);

}

总结

上面的算法使用基本上采用递归的方式,递归的思想在编程中还是很重要的,要注意慢慢培养,另外,要想很好的理解递归的过程,需要学习一下栈帧的概念,可以阅读《深入理解计算机系统》这本书,下篇将对二叉树其他算法进行练习,敬请期待!

参考

二叉树常见面试题目(java实现)

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

推荐阅读更多精彩内容