java数据结构之赫夫曼树

2024-01-20 20:12

本文主要是介绍java数据结构之赫夫曼树,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

        赫夫曼树,又称最优二叉树,是带权路径最短的树。它的基本概念包括路径、路径长度、树的路径长度、权、结点的带权路径长度和树的带权路径长度。其中,路径是从树中一个结点到另一个结点之间的分支构成,路径长度是路径上的分支数目,树的路径长度是从树根到每个结点的路径长度之和。权是赋予树中每个结点的一个有实际意义的数值,比如表示该结点出现的频率等。结点的带权路径长度是从根结点到该结点之间的路径长度与该结点的权的乘积,而树的带权路径长度是树中所有叶子结点的带权路径长度之和。
        构造赫夫曼树的方法是采用赫夫曼算法,即根据给定的权值集合构造出对应的二叉树集合,然后从中选取两棵权值最小的树作为新构造的二叉树的左右子树,新二叉树的根结点的权值为其左右子树根结点的权值之和,最后重复这一步骤,直到只剩下一棵树为止,这棵树便是赫夫曼树。
        赫夫曼树在计算机科学和信息处理领域中有着广泛的应用,比如数据压缩和编码等。其中,赫夫曼编码是一种基于赫夫曼树的编码方式,可以用于数据压缩,其基本原理是给出现频率较高的字符分配较短的编码,给出现频率较低的字符分配较长的编码,从而达到压缩数据的目的。

package com.example.datastructures.tree;import java.util.ArrayList;
import java.util.Collections;
import java.util.List;/*** @author maoyouhua* @version jdk21**          赫夫曼树,又称最优二叉树,是带权路径最短的树。它的基本概念包括路径、路径长度、树的路径长度、权、结点的带权路径长度和树的带权路径长度。*          其中,路径是从树中一个结点到另一个结点之间的分支构成,路径长度是路径上的分支数目,树的路径长度是从树根到每个结点的路径长度之和。*          权是赋予树中每个结点的一个有实际意义的数值,比如表示该结点出现的频率等。*          结点的带权路径长度是从根结点到该结点之间的路径长度与该结点的权的乘积,而树的带权路径长度是树中所有叶子结点的带权路径长度之和。**          构造赫夫曼树的方法是采用赫夫曼算法,即根据给定的权值集合构造出对应的二叉树集合,*          然后从中选取两棵权值最小的树作为新构造的二叉树的左右子树,新二叉树的根结点的权值为其左右子树根结点的权值之和,*          最后重复这一步骤,直到只剩下一棵树为止,这棵树便是赫夫曼树。**          赫夫曼树在计算机科学和信息处理领域中有着广泛的应用,比如数据压缩和编码等。*          其中,赫夫曼编码是一种基于赫夫曼树的编码方式,可以用于数据压缩,*          其基本原理是给出现频率较高的字符分配较短的编码,给出现频率较低的字符分配较长的编码,从而达到压缩数据的目的。*/
public class HuffmanTree {public static Node createHuffmanTree(int[] arr){List<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.getValue() + rightNode.getValue(), leftNode, rightNode);nodes.remove(leftNode);nodes.remove(rightNode);nodes.add(parent);}return nodes.get(0);}public static void preOrder(Node root){if (root != null) {root.preOrder();} else {System.out.println("空树,不能遍历");}}/***             67*          ↙      ↘*      29           38*                ↙         ↘*             15             23*          ↙    ↘        ↙        ↘*        7       8      10         13*                     ↙    ↘*                    4       6*                  ↙    ↘*                 1      3** @param args*/public static void main(String[] args) {int[] arr = {13,7,8,3,29,6,1};Node huffmanTree = createHuffmanTree(arr);preOrder(huffmanTree);}
}/***      实现自然排序接口,方便集合排序*/
class Node implements Comparable<Node>{private Node left;private Node right;private int value;public Node(int value) {this.value = value;}public Node(int value, Node left, Node right) {this.left = left;this.right = right;this.value = value;}public int getValue() {return value;}public void preOrder(){System.out.println(this);if (this.left != null) {this.left.preOrder();}if (this.right != null) {this.right.preOrder();}}@Overridepublic String toString() {return "Node{" +"value=" + value +'}';}@Overridepublic int compareTo(Node o) {return this.value - o.value;}
}

这篇关于java数据结构之赫夫曼树的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/627195

相关文章

JVM 的类初始化机制

前言 当你在 Java 程序中new对象时,有没有考虑过 JVM 是如何把静态的字节码(byte code)转化为运行时对象的呢,这个问题看似简单,但清楚的同学相信也不会太多,这篇文章首先介绍 JVM 类初始化的机制,然后给出几个易出错的实例来分析,帮助大家更好理解这个知识点。 JVM 将字节码转化为运行时对象分为三个阶段,分别是:loading 、Linking、initialization

Spring Security 基于表达式的权限控制

前言 spring security 3.0已经可以使用spring el表达式来控制授权,允许在表达式中使用复杂的布尔逻辑来控制访问的权限。 常见的表达式 Spring Security可用表达式对象的基类是SecurityExpressionRoot。 表达式描述hasRole([role])用户拥有制定的角色时返回true (Spring security默认会带有ROLE_前缀),去

浅析Spring Security认证过程

类图 为了方便理解Spring Security认证流程,特意画了如下的类图,包含相关的核心认证类 概述 核心验证器 AuthenticationManager 该对象提供了认证方法的入口,接收一个Authentiaton对象作为参数; public interface AuthenticationManager {Authentication authenticate(Authenti

Spring Security--Architecture Overview

1 核心组件 这一节主要介绍一些在Spring Security中常见且核心的Java类,它们之间的依赖,构建起了整个框架。想要理解整个架构,最起码得对这些类眼熟。 1.1 SecurityContextHolder SecurityContextHolder用于存储安全上下文(security context)的信息。当前操作的用户是谁,该用户是否已经被认证,他拥有哪些角色权限…这些都被保

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

Java架构师知识体认识

源码分析 常用设计模式 Proxy代理模式Factory工厂模式Singleton单例模式Delegate委派模式Strategy策略模式Prototype原型模式Template模板模式 Spring5 beans 接口实例化代理Bean操作 Context Ioc容器设计原理及高级特性Aop设计原理Factorybean与Beanfactory Transaction 声明式事物

Java进阶13讲__第12讲_1/2

多线程、线程池 1.  线程概念 1.1  什么是线程 1.2  线程的好处 2.   创建线程的三种方式 注意事项 2.1  继承Thread类 2.1.1 认识  2.1.2  编码实现  package cn.hdc.oop10.Thread;import org.slf4j.Logger;import org.slf4j.LoggerFactory

JAVA智听未来一站式有声阅读平台听书系统小程序源码

智听未来,一站式有声阅读平台听书系统 🌟&nbsp;开篇:遇见未来,从“智听”开始 在这个快节奏的时代,你是否渴望在忙碌的间隙,找到一片属于自己的宁静角落?是否梦想着能随时随地,沉浸在知识的海洋,或是故事的奇幻世界里?今天,就让我带你一起探索“智听未来”——这一站式有声阅读平台听书系统,它正悄悄改变着我们的阅读方式,让未来触手可及! 📚&nbsp;第一站:海量资源,应有尽有 走进“智听

【数据结构】——原来排序算法搞懂这些就行,轻松拿捏

前言:快速排序的实现最重要的是找基准值,下面让我们来了解如何实现找基准值 基准值的注释:在快排的过程中,每一次我们要取一个元素作为枢纽值,以这个数字来将序列划分为两部分。 在此我们采用三数取中法,也就是取左端、中间、右端三个数,然后进行排序,将中间数作为枢纽值。 快速排序实现主框架: //快速排序 void QuickSort(int* arr, int left, int rig