【算法数据结构系列】哈夫曼树进阶解读,2024Android架构面试指南

*   [3.3、哈夫曼树的构造](about:blank#33_104)
  • 四、哈夫曼树代码实现

一、树

=================================================================

1.1、什么是树


因为在客观世界里,有许多的事务,存在着细致的划分,比如下图:在我们中国,存储在各个省市的划分。

在这里插入图片描述

那为什么要采取树这种结构呢?

我们可以看到,树这种结构在管理层次里面,它的管理效率更高。因为我们可以根据树的层次去更快的查找数据,比如:中国-湖南-广东-长沙等。树的层次管理具有更高的效率。

1.2、树的定义


树(Tree):N个结点构成的有限集合。

  • 树中有一个称为“根(Root)”的特殊结点。

  • 其余结点可以分为若干个互不相交的树,称为原来结点的“子树”

在这里插入图片描述

从上图中,我们可以看出,A是树的根,B、C、D 都是A的是子树

1.3、树的常用基本术语


在这里插入图片描述

一起来看一下上面的树,我们可以总结出:

  • 结点的度:结点的子树个数。

  • 树的度:树中所有结点中最大的度。

  • 叶结点:度为0的结点。

  • 父结点:所有子树的结点是其子树的根结点的父结点。

  • 子结点:若A是B的父结点,B就是A的子结点。

1.4、树与非树


我们先来看一组错误的树:

在这里插入图片描述

上图中的三种情况,都不能称之为树。为什么呢?

  • 子树是不相交的。

  • 除了根结点之外,每个结点有且只有一个父节点。

  • 一个N个结点的树,只有N-1条边

下面来看一个真正的树:

在这里插入图片描述

二、二叉树

===================================================================

2.1、什么是二叉树


  • 度为2的树(树中所有结点中最大的度)。

  • 子树有左右之分。

在这里插入图片描述

三、Huffman 编码及实现

=============================================================================

3.1、编码问题


给你一段字符串,如何对字符串进行编码,可以使得该字符串的编码存储空间最少

假设一段文本,包含58个字符,并且由以下7个字符构成:a,b,c,d,e,f,g;这7个字符出现的频次不同,如何对这7个字符进行编码,使得总编码空间最小。

在这里插入图片描述

我们一起来分析一下:

在这里插入图片描述

  1. 用等长ASCII编码:58 x 8 = 464位。

  2. 用等长3位编码:58 x 3 = 174位。

  3. 不等长编码:出现频次高的字符用的编码短些,出现频次低的编码长些。

编码长度:10 x 3 + 15 x 2 + 12 x 2 + 5 x 3 + 4 x 4 + 13 x 2 + 5 x 1 = 146位。

3.2、使用二叉树解决编码问题


在这里插入图片描述

使用二叉树进行编码

二叉树左右分支:0、1

根据上图,我们可以发现,编码对应的字符串是:

b 编码 0

f 编码 1

c 编码 10

1 编码 11

在这里插入图片描述

在这里插入图片描述

3.3、哈夫曼树的构造


哈夫曼树:构建一颗二叉树,该树的带权路径长度达到最小,称为最优二叉树,也称为哈夫曼树(Huffman Tree)。

构造方式:每次把权值最小的两颗二叉树合并。左结点权值比右结点小。

四、哈夫曼树代码实现

========================================================================


public class HuffmanTree {

    //节点

    public static class Node<E> {

        E data; //数据

        int weight; //权重

        Node leftChild; //左子节点

        Node rightChild;//右子节点



        public Node(E data, int weight) {

            super();

            this.data = data;

            this.weight = weight;

        }



        public String toString() {

            return "Node[" + weight + ",data=" + data + "]";

        }

    }



    public static void main(String[] args) {

        List<Node> nodes = new ArrayList<Node>();

        //把节点加入至list中

        nodes.add(new Node("a", 10));

        nodes.add(new Node("b", 15));

        nodes.add(new Node("c", 12));

        nodes.add(new Node("d", 3));

        nodes.add(new Node("e", 4));

        nodes.add(new Node("f", 13));

        nodes.add(new Node("g", 1));

        //进行哈夫曼树的构造

        Node root = HuffmanTree.createTree(nodes);

        //打印哈夫曼树

        printTree(root);



    }



    /**

     * 构造哈夫曼树

     *

     * @param nodes

     *            节点集合

     * @return 构造出来的哈夫曼树的根节点

     */

    private static Node createTree(List<Node> nodes) {

        //如果节点node列表中海油2个和2个以上的节点

        while(nodes.size()>1){

            //什么是最小的,list表进行排序,增序的方式, 0,1,

            sort(nodes);//排序方式是增序的

            Node left = nodes.get(0);//权重最小的

            Node right = nodes.get(1);//权重第二小的

            //生成一个新的节点(父节点),父节点的权重为两个子节点的之和

            Node parent = new Node(null,left.weight+right.weight);

            //树的连接,让子节点与父节点进行连接

            parent.leftChild = left;
# 尾声

一转眼时间真的过的飞快。我们各奔东西,也各自踏上了自己的旅途,但是即使多年不见,也因为这份情谊我们依旧如从前那般“亲密”。不忘初心方得始终。加油吧,程序员们,在我看来35岁,40岁从来不是危机,只要永远不要忘记自己为何踏上征程!

#### 最后需要同款资料的,可以 **私信我点击【[学习](https://github.com/a120464/Android-P7/blob/master/Android%E5%BC%80%E5%8F%91%E4%B8%8D%E4%BC%9A%E8%BF%99%E4%BA%9B%EF%BC%9F%E5%A6%82%E4%BD%95%E9%9D%A2%E8%AF%95%E6%8B%BF%E9%AB%98%E8%96%AA%EF%BC%81.md)】**我愿意分享给你!

为了让更多在学习中或者最近要准备面试的朋友们看到这篇文章,希望你们能多多评论,点赞+转发!

**再次感谢所有给我提供过题目的朋友们,感谢一路有你!**
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值