[061] Java “哈夫曼树”

1、哈夫曼树

  • Huffman Tree:给定N个权值作为N个叶子节点,构造一棵二叉树,若该树的带权路径长度达到最小,称这样的二叉树为最优二叉树,也称为哈夫曼树。、
    • 1、路径:在一棵树中,从一个节点往下可以达到的孩子或孙子节点之间的通路,称为路径;
    • 2、路径长度:路径分支(叉)的次数称为路径长度,则从根结点到第L层结点的路径长度为 L-1;
    • 3、节点的权:节点被赋予的值;
    • 4、带权路径长度:路径长度与节点权值的乘积;
    • 5、树的带权路径:所有叶子节点的带权路径长度之和,记为WPL(weight path length)
    • 6、哈夫曼树:WPL最小的一种二叉树,一般权值越大,离根节点越近;
  • 构建哈夫曼树的步骤:
    • 1、遍历arr数组 ;
    • 2、将arr的每个元素构成一个节点Node,Node含权值域即左右指针域;
    • 3、将Node放入ArrayList中,做排序处理;
    • 4、取出权值最小第二小的两个节点(或二叉树),构建成一棵新的二叉树;
    • 5、从ArrayList中删除刚才使用过的两个节点(或二叉树);
    • 6、将新构建的二叉树放入ArrayList;
    • 7、重复 2-6,直到ArrayList中的元素只剩下一个;

2、Java代码 -- 构建哈夫曼树

package DataStructures.Tree;

import java.util.ArrayList;
import java.util.Collections;

/**
 * @author yhx
 * @date 2020/10/16
 */
public class HuffmanTree {
    public static void main(String[] args) {
        int[] arr = {13, 7, 8, 3, 29, 6, 1};
        Node root = createHuffmanTree(arr);
        System.out.println("哈夫曼树前序遍历结果为:");
        preOrder(root);
    }

    /**
     * 创建哈夫曼树
     * 1、遍历arr数组
     * <p>
     * 2、将arr的每个元素构成一个节点Node
     * 3、将Node放入ArrayList中,做排序处理
     * 4、取出权值最小和第二小的两个节点(或二叉树),构建成一棵新的二叉树
     * 5、从ArrayList中删除刚才使用过的两个节点(或二叉树)
     * 6、将新构建的二叉树放入ArrayList
     * <p>
     * 7、重复 2-6,直到ArrayList中的元素只剩下一个
     *
     * @param arr 原始数组
     * @return 返回创建好的哈夫曼树的根节点
     */
    public static Node createHuffmanTree(int[] arr) {
        ArrayList<Node> nodes = new ArrayList<>();
        for (int value : arr) {
            nodes.add(new Node(value));
        }
        while (nodes.size() > 1) {
            // 升序排序
            Collections.sort(nodes);

            // 取出权值最小和第二小的节点(或二叉树)
            Node leftNode = nodes.get(0);
            Node rightNode = nodes.get(1);
            // 构建成一棵新的二叉树,这棵二叉树的父节点权值等于两个子节点权值之和
            Node parent = new Node(leftNode.value + rightNode.value);
            parent.left = leftNode;
            parent.right = rightNode;

            // 删除刚才使用过的两个二叉树
            nodes.remove(leftNode);
            nodes.remove(rightNode);

            // 将新构建的二叉树放入ArrayList
            nodes.add(parent);
        }
        // 返回哈夫曼树的root节点
        return nodes.get(0);
    }

    /**
     * 前序遍历
     *
     * @param root 根节点
     */
    public static void preOrder(Node root) {
        if (root != null) {
            root.preOrder();
        } else {
            System.out.println("空树,不能遍历");
        }
    }
}


/**
 * 节点类
 * comparable接口:使继承它的类比较大小
 */
class Node implements Comparable<Node> {
    /**
     * value:节点权值
     * left:指向左子节点
     * right:指向右子节点
     */
    int value;
    Node left;
    Node right;

    /**
     * 构造器
     *
     * @param value 节点权值
     */
    public Node(int value) {
        this.value = value;
    }

    @Override
    public String toString() {
        return "Node [value = " + value + "]";
    }

    @Override
    public int compareTo(Node o) {
        // 升序排序
        return this.value - o.value;
        // 降序排序
        // return -(this.value - o.value);
    }

    /**
     * 前序遍历方法
     */
    public void preOrder() {
        System.out.println(this);
        if (this.left != null) {
            this.left.preOrder();
        }
        if (this.right != null) {
            this.right.preOrder();
        }
    }
}

3、运行结果

哈夫曼树前序遍历结果为:
Node [value = 67]
Node [value = 29]
Node [value = 38]
Node [value = 15]
Node [value = 7]
Node [value = 8]
Node [value = 23]
Node [value = 10]
Node [value = 4]
Node [value = 1]
Node [value = 3]
Node [value = 6]
Node [value = 13]

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
数据结构关于二叉树的建立遍历以及应用二叉树进行编解码 实验要求 必做部分 1. 小明会按照前序的方式输入一棵二叉树。例如,输入$ACG##H##D##BE#I##F##的话,代表了下面这棵树: 2. 请分别按照前序、中序、后序输出这棵树。 选做部分 背景 在影视剧中,我们经常会看到二战期间情报人员使用电报哒哒哒地发送信息,发送电报所使用的编码叫做摩尔斯电码(或者叫做摩斯密码)。甚至在现代,SOS仍然是国际通用的求救信号之一,其“三短、三长、三短”同样是摩斯密码的编码方式。 摩斯密码使用若干个“点”和“划”来表示一个字母,字母和字母之间使用短暂的停顿来表示。例如,一种常见的编码方式为: 字母 摩斯密码 字母 摩斯密码 A .- E . B -... F ..-. C -.-. G --. D -.. H .... 实际上,一个摩斯密码本可以使用一棵二叉树来存储: 上图表示,从根节点start开始,遇到一个点(Dot)就访问它的左子树节点,遇到一个划(Dash)就访问它的右子树节点。例如,三个点...代表了S,三个划---代表了O。所以SOS的摩斯密码是... --- ...(中间用空格隔开,表示短暂的停顿)。再比如,爱疯手机有一种来电铃声的节奏为“哇哇哇 哇-哇- 哇哇哇”,这其实表示的是…… 现在,小明想在课上偷偷跟你传纸条,但又不想被其他同学看到内容。因此他跟你约定,每次给你传纸条时都使用摩斯密码来编写。至于密码本,当然不能使用国际通用的,他会在课前告诉你密码本的内容。然而小明发现,每次写纸条、读纸条都不是很方便,所以他想让你做个程序来自动编码/解码你们的摩斯密码。 题目要求 首先,小明输入的那棵二叉树,代表了你们在这堂课上要使用的摩斯密码本。例如,输入$ACG##H##D##BE#I##F##的话,代表了下面这棵树: 第一个字母$是什么并不重要,因为它只是代表了根节点,而根节点在我们的摩斯电码中并不代表一个字符,仅仅代表“start”。 读入密码本后,请记得按照前序、中序、后序输出这棵树。 然后,小明会输入一个数,代表接下来输入的是明文还是摩斯电码。输入0表示接下来他会输入明文,输入1表示接下来输入的是摩斯电码,输入-1程序退出。 1. 如果输入的是0,代表接下来要输入的是明文。程序接受一个字符串,根据字符串中每一个字母输出对应的摩斯电码,用空格隔开。例如如果小明输入“BED”,则程序应该输出“-空格-.空格.-”。如果遇到密码本中没有的字符,则输出“输入有误”。 2. 如果输入的是1,则表示接下来要输入摩斯电码。小明首先会输入一个数字N,代表有几个电码的输入,例如输入4代表之后会输入4个电码(即这个单词有四个字母)。随后输入空格分割的电码,例如, .. . -- -. 程序需要根据摩斯电码解读出明文单词并输出,例如上面的电码表示“CAFE”。如果遇到密码本中没有的编码,则输出“输入有误”。 输入输出样例 必做部分: 请输入二叉树: $ACG##H##D##BE#I##F## 前序遍历:$ACGHDBEIF 中序遍历:GCHAD$EIBF 后序遍历:GHCDAIEFB$ 选做部分: 请选择(0为明文,1为电码,-1退出):0 请输入明文:BED 摩斯电码为:- -. .- 请选择(0为明文,1为电码,-1退出):1 请输入电码个数:4 请输入电码:.. . -- -. 明文为:CAFE 请选择(0为明文,1为电码,-1退出):-1 // 程序结束
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值