java-贪心算法

1. 霍夫曼编码(Huffman Coding)

描述
霍夫曼编码是一种使用变长编码表对数据进行编码的算法,由David A. Huffman在1952年发明。它是一种贪心算法,用于数据压缩。霍夫曼编码通过构建一个二叉树(霍夫曼树),树中的每个叶子节点代表一个字符,树的权重表示字符出现的频率。构建树的过程中,总是将两个权重最小的节点合并。

Java案例

import java.util.PriorityQueue;

public class HuffmanCoding {
    static class Node {
        char data;
        int frequency;
        Node left, right;

        Node(char data, int frequency) {
            this.data = data;
            this.frequency = frequency;
            left = right = null;
        }
    }

    // 构建霍夫曼树
    static Node buildHuffmanTree(char data[], int frequency[]) {
        PriorityQueue<Node> minHeap = new PriorityQueue<>((a, b) -> a.frequency - b.frequency);
        for (int i = 0; i < data.length; i++) {
            minHeap.add(new Node(data[i], frequency[i]));
        }

        while (minHeap.size() > 1) {
            Node x = minHeap.poll();
            Node y = minHeap.poll();
            Node f = new Node('\0', x.frequency + y.frequency);
            f.left = x;
            f.right = y;
            minHeap.add(f);
        }

        return minHeap.poll();
    }

    // 打印霍夫曼编码
    static void printCodes(Node root, String s) {
        if (root.left == null && root.right == null && Character.isLetter(root.data)) {
            System.out.println(root.data + ": " + s);
            return;
        }

        printCodes(root.left, s + "0");
        printCodes(root.right, s + "1");
    }

    public static void main(String[] args) {
        char data[] = {'a', 'b', 'c', 'd', 'e', 'f'};
        int frequency[] = {5, 9, 12, 13, 16, 45};
        Node root = buildHuffmanTree(data, frequency);
        printCodes(root, "");
    }
}

2. 活动选择问题(Activity Selection Problem)

描述
活动选择问题是一个经典的贪心算法问题,给定一系列活动,每个活动都有开始时间和结束时间,目标是选择最大数量的互不重叠的活动。

Java案例

import java.util.Arrays;

public class ActivitySelection {
    public static void selectActivities(int start[], int end[]) {
        Arrays.sort(end); // 按结束时间排序
        int i = 0, activities = 0;
        for (int j = 1; j < end.length; j++) {
            if (start[i] < end[j]) { // 如果当前活动的开始时间小于前一个活动的结束时间
                activities++;
                i = j; // 更新当前活动的索引
            }
        }
        System.out.println("Total activities selected: " + activities);
    }

    public static void main(String[] args) {
        int start[] = {1, 3, 0, 5, 8, 5};
        int end[] = {2, 4, 6, 7, 9, 9};
        selectActivities(start, end);
    }
}

3. 硬币找零问题(Coin Change Problem)

描述
硬币找零问题是一个典型的贪心算法问题,给定不同面额的硬币和一个总金额,目标是使用最少数量的硬币凑成总金额。

Java案例

public class CoinChange {
    public static int coinChange(int[] coins, int amount) {
        int[] dp = new int[amount + 1];
        Arrays.fill(dp, amount + 1); // 初始化为最大值
        dp[0] = 0; // 0元需要0个硬币

        for (int i = 1; i <= amount; i++) {
            for (int coin : coins) {
                if (i >= coin && dp[i - coin] != amount + 1) {
                    dp[i] = Math.min(dp[i], dp[i - coin] + 1);
                }
            }
        }

        return dp[amount] > amount ? -1 : dp[amount];
    }

    public static void main(String[] args) {
        int[] coins = {1, 2, 5};
        int amount = 11;
        int result = coinChange(coins, amount);
        if (result != -1) {
            System.out.println("Minimum number of coins required: " + result);
        } else {
            System.out.println("Not possible to make the given amount");
        }
    }
}

这些案例展示了贪心算法在不同问题中的应用,包括霍夫曼编码、活动选择问题和硬币找零问题。贪心算法通过在每一步选择局部最优解来寻找全局最优解。

🌐 霍夫曼编码和哈夫曼编码有什么区别?

霍夫曼编码(Huffman Coding)和哈夫曼编码(Huffman Coding)实际上是同一个概念,只是翻译名称上的不同。在中文文献和技术社区中,这两种叫法都被广泛使用,但它们指的是同一种编码方法。

霍夫曼编码是一种广泛使用的无损数据压缩算法,由David A. Huffman在1952年发明。它是一种贪心算法,通过构建一个二叉树(霍夫曼树)来为输入字符分配可变长度的编码。树的叶子节点代表输入字符,而树的权重表示字符出现的频率。构建树的过程中,总是将两个权重最小的节点合并,直到构建出完整的霍夫曼树。然后,通过遍历树来为每个字符分配一个唯一的二进制编码,频率高的字符会被分配较短的编码,而频率低的字符则会被分配较长的编码。

因此,当你看到“霍夫曼编码”和“哈夫曼编码”时,可以认为它们是同一个算法的不同叫法。在英文中,这个算法统一被称为“Huffman Coding”。
在这里插入图片描述

🤔 霍夫曼编码在哪些场景下应用效果最好?

霍夫曼编码(Huffman Coding)在以下场景下应用效果最好:

  1. 文件压缩
    霍夫曼编码广泛应用于文件压缩领域,特别是在存储文件时,可以通过压缩来节省磁盘空间。虽然文件压缩后需要解压缩才能使用,但这通常是一次性的操作,因此空间和时间的权衡是可接受的。例如,在文件系统中存储不常使用的文件时,使用霍夫曼编码进行压缩可以显著减少存储空间的需求。

  2. 数据通信
    在数据通信领域,霍夫曼编码可以减少通信和I/O开销,特别是在高速流处理中。压缩原始模拟数据并在后处理前解压缩可以显著提高工作效率。

  3. 图像和视频压缩
    霍夫曼编码是许多图像和视频压缩标准的一部分,如JPEG。它用于对图像和视频数据进行无损压缩,是这些应用中不可或缺的一部分。

  4. 大规模科学数据集
    霍夫曼编码也被广泛用于压缩大型科学数据集,这些数据集对数据保真度有很高的要求。它在这些应用中表现出色,因为它可以有效地压缩数据,同时保持高数据保真度。

  5. 文本编码
    霍夫曼编码用于文本编码,特别是在不同字符使用频率差异较大的情况下。它通过为更频繁出现的字符分配较短的编码来实现空间效率,这在英语文档等文本中尤其有效,因为不同字母的使用频率差异很大。

  6. 通用无损压缩软件或库
    霍夫曼编码是许多通用无损压缩软件或库(如GZIP、Zstd和Blosc)中的关键步骤,它提供了一种有效的熵编码算法。

  7. 错误有界损失压缩
    在一些需要错误有界损失压缩的应用中,霍夫曼编码也被广泛使用,例如SZ和MGARD压缩器,这些压缩器在压缩大数据集时非常有效。

综上所述,霍夫曼编码在需要有效压缩数据以节省空间或减少传输时间的场景中效果最好,尤其是在字符使用频率不均的文本数据、图像和视频数据以及大规模科学数据集的压缩中。

📈 霍夫曼编码在减少网络带宽方面有哪些优势?

霍夫曼编码在减少网络带宽方面的优势主要体现在以下几个方面:

  1. 高效压缩
    霍夫曼编码为每个字符分配一个基于其出现频率的变长编码,高频字符使用较短的编码,低频字符使用较长的编码。这种优化可以显著减少数据的整体大小,从而减少网络传输所需的带宽。

  2. 适应性
    霍夫曼编码是自适应的,可以根据数据的实际内容动态构建最优编码树。这意味着它能够针对特定的数据集进行优化,与固定编码方案相比,能够更有效地压缩数据。

  3. 无损压缩
    霍夫曼编码是一种无损压缩算法,这意味着压缩和解压后的数据与原始数据完全相同。这对于必须确保数据完整性的应用场景(如文件传输、视频流等)至关重要。

  4. 广泛支持
    霍夫曼编码被广泛支持和实现在各种压缩工具和协议中,如HTTP压缩、ZIP文件格式、JPEG图像压缩等。这种广泛的支持意味着它可以很容易地集成到现有的网络通信系统中。

  5. 减少延迟
    通过减少数据的大小,霍夫曼编码可以减少数据在网络上的传输时间,从而降低延迟,特别是在高延迟网络环境中。

  6. 节省成本
    减少网络带宽的使用可以直接降低网络服务提供商的成本,尤其是在需要支付流量费用的场景下,如移动数据服务。

  7. 提高吞吐量
    在带宽受限的网络中,更有效地压缩数据可以提高网络的吞吐量,允许在同一带宽内传输更多的数据。

  8. 支持流式传输
    霍夫曼编码支持流式压缩和解压,这意味着数据可以边传输边解压,而不需要等待整个文件传输完成,这对于流媒体应用(如视频会议和在线视频流)非常有用。

  9. 易于实现
    霍夫曼编码算法相对简单,易于实现,并且有许多现成的库和工具可供使用,这使得它能够快速部署在各种网络应用中。

  10. 可扩展性
    霍夫曼编码可以应用于不同规模的数据,从小型文本文件到大型视频文件,都能有效地减少所需的网络带宽。

综上所述,霍夫曼编码在减少网络带宽方面的优势使其成为网络数据传输中不可或缺的技术之一。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/920469.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

RK3568平台(中断篇)ARM中断流程

一.ARM 处理器程序运行的过程 ARM芯片属于精简指令集计算机 (RISC: Reduced Instruction Set Computing),它所用的指令比较简单,有如下特点: ① 对内存只有读、写指令 ② 对于数据的运算是在CPU内部实现 ③ 使用RISC指令的CPU复杂度小一点,易于设计 比如对于 a=a+b 这…

SpringBoot与MongoDB深度整合及应用案例

SpringBoot与MongoDB深度整合及应用案例 在当今快速发展的软件开发领域&#xff0c;NoSQL数据库因其灵活性和可扩展性而变得越来越流行。MongoDB&#xff0c;作为一款领先的NoSQL数据库&#xff0c;以其文档导向的存储模型和强大的查询能力脱颖而出。本文将为您提供一个全方位…

项目实战(webshop)

一、搭建靶场webshop(www.example1.com) 二、信息收集 1、获取IP winR→cmd→ping www.example1.com→显示出ip(192.168.51.128) 注&#xff1a;TTL为IP包的生存时间&#xff0c;拿到TTL我们可以大概的判断一下目标机器的操作系统&#xff0c;但不是很准确&#xff0c;因为…

初识算法 · 分治(3)

目录 前言&#xff1a; 归并排序 题目解析 算法原理 算法编写 求逆序对总数 题目解析 算法原理 算法编写 前言&#xff1a; ​本文的主题是分治&#xff0c;通过两道题目讲解&#xff0c;一道是归并排序&#xff0c;一道是求逆序对。 链接分别为&#xff1a; 912. 排…

MyBatis——#{} 和 ${} 的区别和动态 SQL

1. #{} 和 ${} 的区别 为了方便&#xff0c;接下来使用注解方式来演示&#xff1a; #{} 的 SQL 语句中的参数是用过 ? 来起到类似于占位符的作用&#xff0c;而 ${} 是直接进行参数替换&#xff0c;这种直接替换的即时 SQL 就可能会出现一个问题 当传入一个字符串时&#xff…

学习threejs,导入FBX格式骨骼绑定模型

&#x1f468;‍⚕️ 主页&#xff1a; gis分享者 &#x1f468;‍⚕️ 感谢各位大佬 点赞&#x1f44d; 收藏⭐ 留言&#x1f4dd; 加关注✅! &#x1f468;‍⚕️ 收录于专栏&#xff1a;threejs gis工程师 文章目录 一、&#x1f340;前言1.1 ☘️THREE.FBXLoader fbx 模型加…

【腾讯云产品最佳实践】腾讯云CVM入门技术与实践:通过腾讯云快速构建云上应用

目录 前言 什么是腾讯云CVM&#xff1f; 腾讯云CVM的技术优势 基于最佳技术实践&#xff0c;使用腾讯云CVM搭建应用 1. 开通CVM实例 2. 连接CVM实例 3. 配置Web环境 4. 部署PHP应用 腾讯云CVM行业应用案例&#xff1a;电商平台的双十一攻略 1. 弹性伸缩解决高并发问题…

mongodb多表查询,五个表查询

需求是这样的&#xff0c;而数据是从mysql导入进来的&#xff0c;由于mysql不支持数组类型的数据&#xff0c;所以有很多关联表。药剂里找药物&#xff0c;需要药剂与药物的关联表&#xff0c;然后再找药物表。从药物表里再找药物与成分关联表&#xff0c;最后再找成分表。 这里…

STL中vector实现——简单易懂版

本章内容 模拟实现 vector 的部分重要功能 1.迭代器的引入1.1 之前写法1.2 STL库中的写法 2.默认成员函数2.1构造与拷贝构造2.2拷贝赋值2.3析构函数 3.增删查改功能3.1插入3.2删除 4.为什么STL中vector没有find函数&#xff1f;5.&#x1f525;&#x1f525;迭代器失效场景&am…

Springboot + vue 健身房管理系统项目部署

1、前言 ​ 许多人在拿到 Spring Boot 项目的源码后&#xff0c;不知道如何运行。我以 Spring Boot Vue 健身房管理系统的部署为例&#xff0c;详细介绍一下部署流程。大多数 Spring Boot 项目都可以通过这种方式部署&#xff0c;希望能帮助到大家。 ​ 2、项目查看 ​ 首…

NuGet如何支持HTTP源

今天是2024年11月21号&#xff0c;最近更新了VisualStudio后发现HTTP的包源已经默认禁止使用了&#xff0c;生成时会直接报错。如下图&#xff1a; 官方也明确指出了要想使用HTTP包源的解决办法&#xff0c;这里就简单总结一下。 一、全局配置 1、全局NuGet包的配置文件路径在…

SpringBoot学习记录(四)之分页查询

SpringBoot学习记录&#xff08;四&#xff09;之分页查询 一、业务需求1、基本信息2、请求参数3、相应数据 二、传统方式分页三、使用PageHelper分页插件 一、业务需求 根据条件进行员工数据的条件分页查询 1、基本信息 请求路径&#xff1a; /emps 请求方式&#xff1a; …

JavaParser如何获取方法的返回类型

使用JavaParser 如何获取一个Java类中的某个方法的返回类型呢&#xff1f; 假如有一个如下的简单的Java 类&#xff1a; /*** Copyright (C) Oscar Chen(XM):* * Date: 2024-11-21* Author: XM*/ package com.osxm.ai.sdlc.codeparse.codesample;public class MyClass {public…

2024亚太杯国际赛C题宠物预测1234问完整解题思路代码+成品参考文章

中国宠物业发展趋势及预测模型 一、问题背景与研究目标 近年来&#xff0c;中国宠物业经历了快速发展&#xff0c;特别是在城市化进程加快、人口结构变化和消费水平提升的背景下&#xff0c;宠物作为家庭成员的角色变得愈发重要。根据相关数据&#xff0c;中国宠物数量&#…

Java实现离线身份证号码OCR识别

最近公司要求做离线身份证OCR功能&#xff0c;找了一圈总算是找到了&#xff0c;在这里对文档做个整理&#xff0c;方便后来者&#xff0c;感谢码龄23年博主的分享 系统&#xff1a;Windows11&#xff0c;红旗Linux Asianux8.1 文档中Linux全root用户操作&#xff1b;需先安装…

Gradle核心概念总结

这部分内容主要根据 Gradle 官方文档整理&#xff0c;做了对应的删减&#xff0c;主要保留比较重要的部分&#xff0c;不涉及实战&#xff0c;主要是一些重要概念的介绍。 Gradle 这部分内容属于可选内容&#xff0c;可以根据自身需求决定是否学习&#xff0c;目前国内还是使用…

鸿蒙网络编程系列50-仓颉版TCP回声服务器示例

1. TCP服务端简介 TCP服务端是基于TCP协议构建的一种网络服务模式&#xff0c;它为HTTP&#xff08;超文本传输协议&#xff09;、SMTP&#xff08;简单邮件传输协议&#xff09;等高层协议的应用程序提供了可靠的底层支持。在TCP服务端中&#xff0c;服务器启动后会监听一个或…

第5-1节:SpringBoot对SpringMVC的自动配置

我的后端学习大纲 SpringBoot学习大纲 1、SpringBoot对SpringMVC自动配置概览

Emacs进阶之插入时间信息(一百六十三)

简介&#xff1a; CSDN博客专家、《Android系统多媒体进阶实战》一书作者 新书发布&#xff1a;《Android系统多媒体进阶实战》&#x1f680; 优质专栏&#xff1a; Audio工程师进阶系列【原创干货持续更新中……】&#x1f680; 优质专栏&#xff1a; 多媒体系统工程师系列【…

嵌入式实验报告:家用计时器

实验目的和要求 1、实验目的 掌握STM32串口通信原理。学习编程实现STM32的UART通信掌握STM32中断程序设计流程。熟悉STM32固件库的基本使用。熟悉STM32定时器中断设计流程。2、实验要求 设计一个家用计时器,其功能如下: 利用串口设置计时时间,格式:XX:XX:X 例如01:59:…