Java玩转《啊哈算法》排序之桶排序

过去心不可得,现在心不可得,未来心不可得

目录在这里

  • 楔子
  • 代码地址
  • 桶排序
  • 代码
    • 核心部分
      • 优缺点
    • 完整代码
    • 演示
  • 升级版
    • 核心代码
    • 完整代码
    • 演示

楔子

大家好!本人最近看了下《啊哈算法》,写的确实不错,生动形象又有趣(我没收钱,确实如此 )。

但对我来说,稍显遗憾的是,书籍代码是c语言,而不是本人常用的Java。

那就弥补遗憾,说干就干,把这本书的示例语言用java给翻译一遍!!!

于是就有了本篇博客,当然这只是第一篇,主要是讲解桶排序。
在这里插入图片描述

没有买纸质书的童鞋也甭担心,电子版的下载链接已经放到下方了,自己下载去吧!!!

链接:https://pan.baidu.com/s/1imxiElcCorw2F-HJEnB-PA?pwd=jmgs
提取码:jmgs

不过还是建议有条件的同学可以买下纸质书,尊重一下作者的劳动成果。

代码地址

本文代码已开源:

git clone https://gitee.com/guqueyue/my-blog-demo.git

请切换到gitee分支,

然后查看aHaAlgorithm模块下的src/main/java/com/guqueyue/aHaAlgorithm/chapter_1_Sort即可!

桶排序

算法学习千千万,排序是块敲门砖!!!

国内算法学习似乎有个不成文的规定,想学算法,先学排序。而桶排序可以说是排序算法中最简单的算法了。

在这里插入图片描述

桶排序的核心原理非常简单:

遍历需要排序的元素集合,用一个数组表示。数组的一个个连续的空间作为一个个桶,索引为元素,而索引对应的值为元素个数。

相当于把元素放到对应的一个一个桶里面,所以叫桶排序。
在这里插入图片描述

而因为数组的索引是连续的,所以遍历数组索引就能得到一个升序的元素集合。如果索引对应的值为0,说明该元素不存在。

代码

核心部分

	/*** @Description 桶排序* @Param [scoreArr]* @return int[]**/private static int[] bucketSort(int[] scoreArr) {// 11为数据范围的大小int[] bucket = new int[11];// 用于返回排序后的数组int[] result = new int[scoreArr.length];// 入桶,计数for (int num : scoreArr) {bucket[num]++;}// 根据桶的索引以及计数的次数,生成排序后的数组 - 如果需要降序,倒序遍历数组即可int k = 0;for (int i = 0; i < bucket.length; i++) { // 遍历每个桶for (int j = 0; j < bucket[i]; j++) { // 遍历桶里面的元素result[k++] = i;}}return result;}

优缺点

通过上文的讲解以及核心代码,我们不难得出桶排序具有以下的优缺点:

  • 优点:
    1. 简单
    2. 速度快。时间复杂度为:O(m+n), 其中m为排序数组的大小,n为桶的大小。
  • 缺点:
    1. 占用空间。因为桶的长度取决于元素的取值范围,元素的取值范围越大,越占用空间。
    2. 有使用局限,只能对整数进行排序。若元素中存在小数无法使用桶排序,因为数组的索引不能为小数。

完整代码

package com.guqueyue.aHaAlgorithm.chapter_1_Sort;import java.util.Arrays;
import java.util.Scanner;/*** @Author: guqueyue* @Description: 桶排序* @Date: 2024/1/8**/
public class BucketSort {public static void main(String[] args) {// 获取分数数组int[] scoreArr = getScoreArr();System.out.println("输入的数组为: " + Arrays.toString(scoreArr));// 桶排序int[] result = bucketSort(scoreArr);System.out.println("排序后:" + Arrays.toString(result));}/*** @Description 桶排序* @Param [scoreArr]* @return int[]**/private static int[] bucketSort(int[] scoreArr) {// 11为数据范围的大小int[] bucket = new int[11];// 用于返回排序后的数组int[] result = new int[scoreArr.length];// 入桶,计数for (int num : scoreArr) {bucket[num]++;}// 根据桶的索引以及计数的次数,生成排序后的数组 - 如果需要降序,倒序遍历数组即可int k = 0;for (int i = 0; i < bucket.length; i++) { // 遍历每个桶for (int j = 0; j < bucket[i]; j++) { // 遍历桶里面的元素result[k++] = i;}}return result;}/*** @Description 获取分数数组* @Param []* @return int[]**/private static int[] getScoreArr() {Scanner scanner = new Scanner(System.in);System.out.print("请输入数组长度:");int n = scanner.nextInt();int[] scoreArr = new int[n];for (int i = 0; i < n; i++) {System.out.printf("请输入第%d个数(范围:0-10),然后按回车: ", i+1);scoreArr[i] = scanner.nextInt();}return scoreArr;}
}

演示

运行代码,控制台输入可得:
在这里插入图片描述

升级版

正如作者所说,上文演示的只是一个简易版的桶排序算法。

那如果需要输入多个学生的姓名和分数,再根据学生的分数排名由高到低输出学生的姓名,这样要怎么做呢?

作者这里并没有给出答案,我们来扩展一下,首先创建一个学生类:

package com.guqueyue.aHaAlgorithm.entity;import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;/*** @Author: guqueyue* @Description: 学生类* @Date: 2024/1/9**/
//lombok插件的注解
@Data // 若未使用lombok插件,请自行生成getter、setter以及toString方法
@NoArgsConstructor // 若未使用lombok插件,请自行生成无参构造方法
@AllArgsConstructor // 若未使用lombok插件,请自行生成有参构造方法
public class Student {private String name; // 姓名private Integer score; // 分数
}

核心代码

同理,我们使用桶排序,得到一个通过分数降序排序的学生数组:

	/*** @Description 桶排序 - 通过分数排序学生数组* @Param [scoreArr]* @return com.guqueyue.aHaAlgorithm.entity.Student[]**/private static Student[] bucketSort(Student[] scoreArr) {Student[] result = new Student[scoreArr.length];// 桶排序,将分数入桶int[] bucket = new int[101];for (Student student : scoreArr) {bucket[student.getScore()]++;}int k = 0;for (int i = 100; i >= 0; i--) { // 倒序遍历桶if (bucket[i] > 0) {for (Student student : scoreArr) { // 遍历学生数组,将符合当前桶的分数的学生放入数组if (student.getScore() == i) {result[k++] = student;}}}}return result;}

完整代码

package com.guqueyue.aHaAlgorithm.chapter_1_Sort;import com.guqueyue.aHaAlgorithm.entity.Student;import java.util.*;/*** @Author: guqueyue* @Description: 桶排序 - 通过分数排序学生数组* @Date: 2024/1/9**/
public class BucketSort2 {public static void main(String[] args) {Student[] scoreArr = getStudentArr(); // 获取学生数组System.out.println("输入的数组为:" + Arrays.toString(scoreArr));Student[] result = bucketSort(scoreArr);System.out.println("排序后的数组为: " + Arrays.toString(result));System.out.print("学生排名为: ");for (Student student : result) {System.out.print(student.getName() + " ");}System.out.println();}/*** @Description 桶排序 - 通过分数排序学生数组* @Param [scoreArr]* @return com.guqueyue.aHaAlgorithm.entity.Student[]**/private static Student[] bucketSort(Student[] scoreArr) {Student[] result = new Student[scoreArr.length];// 桶排序,将分数入桶int[] bucket = new int[101];for (Student student : scoreArr) {bucket[student.getScore()]++;}int k = 0;for (int i = 100; i >= 0; i--) { // 倒序遍历桶if (bucket[i] > 0) {for (Student student : scoreArr) { // 遍历学生数组,将符合当前桶的分数的学生放入数组if (student.getScore() == i) {result[k++] = student;}}}}return result;}/*** @Description 桶排序优化版 - 通过分数排序学生数组* @Param [scoreArr]* @return com.guqueyue.aHaAlgorithm.entity.Student[]**/private static Student[] bucketSort2(Student[] scoreArr) {// 1.构建 分数 -> 人名集合 映射集Map<Integer, List<Student>> dict = new HashMap<>();for (Student student : scoreArr) {Integer score = student.getScore();List<Student> studentList = new ArrayList<>();if (dict.containsKey(score)) {studentList = dict.get(score);}studentList.add(student);dict.put(score, studentList);}Student[] result = new Student[scoreArr.length];// 桶排序int[] bucket = new int[101];for (Student student : scoreArr) {bucket[student.getScore()]++;}int k = 0;for (int i = 100; i >= 0; i--) {if (bucket[i] > 0) { // 如果有List<Student> students = dict.get(i);if (students != null && students.size() > 0) {for (Student student : students) {result[k++] = student;}}}}return result;}/*** @Description 获取学生数组* @Param []* @return com.guqueyue.aHaAlgorithm.entity.Student[]**/private static Student[] getStudentArr() {Scanner scanner = new Scanner(System.in);System.out.print("请输入学生数量:");int n = scanner.nextInt();Student[] students = new Student[n];for (int i = 0; i < n; i++) {Student student = new Student();System.out.printf("请输入第%d个学生的姓名:", i+1);student.setName(scanner.next());System.out.printf("请输入第%d个学生的分数(0-100):", i+1);student.setScore(scanner.nextInt());students[i] = student;}return students;}
}

演示

运行代码,控制台输入,可得:
在这里插入图片描述

我们下期博客再见!

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

【多线程】多线程安全,为什么不安全,要怎么做保证其安全,实例

多线程安全是指在多线程环境下&#xff0c;对共享的数据进行操作时&#xff0c;保证这些操作能够正确、稳定地执行&#xff0c;不会导致数据不一致、异常等问题。多线程不安全通常是因为多个线程同时访问、修改了共享的数据&#xff0c;导致一些不可预料的结果。 一、多线程安…

训练自己的ai模型(一)学习笔记与项目实操

ai模型大火&#xff0c;作为普通人&#xff0c;我也想做个自己的ai模型 训练自己的ai模型通常需要接下来的的六步 一、 收集和准备数据集&#xff1a;需要收集和准备一个数据集&#xff0c;其中包含想要训练模型的数据。这可能需要一些数据清理和预处理&#xff0c;以确保数据…

Optional的使用(代替if判断是否为空)

Optional 前言 我的使用 package yimeng;import com.ruoyi.RuoYiApplication; import com.ruoyi.common.core.domain.entity.SysUser; import org.junit.jupiter.api.Test; import org.springframework.boot.test.context.SpringBootTest; import java.util.*;SpringBootTes…

C语言-算法-拓扑排序

【模板】拓扑排序 / 家谱树 题目描述 有个人的家族很大&#xff0c;辈分关系很混乱&#xff0c;请你帮整理一下这种关系。给出每个人的后代的信息。输出一个序列&#xff0c;使得每个人的后辈都比那个人后列出。 输入格式 第 1 1 1 行一个整数 N N N&#xff08; 1 ≤ N …

OpenCV第 2 课 OpenCV 环境搭建

文章目录 第 2 课 OpenCV 环境搭建1.安装 Numpy2.从 Ubuntu 存储库安装 OpenCV3.验证 OpenCV 安装 第 2 课 OpenCV 环境搭建 1.安装 Numpy 每一张图像都有很多个像素点&#xff0c;这也导致了程序中会涉及大量的数组处理。Numpy 是一个 Python 的拓展库&#xff0c;它对多维数…

仿真机器人-深度学习CV和激光雷达感知(项目2)day7【ROS关键组件】

文章目录 前言Launch 文件了解 XML 文件Launch 文件作用Launch 文件常用标签实例--作业1的 Launch 文件TF Tree介绍发布坐标变换--海龟例程获取坐标变换--海龟自动跟随例程rqt_工作箱前言 💫你好,我是辰chen,本文旨在准备考研复试或就业 💫本文内容是我为复试准备的第二个…

LeetCode 每日一题 Day 51 - 53

670. 最大交换 给定一个非负整数&#xff0c;你至多可以交换一次数字中的任意两位。返回你能得到的最大值。 示例 1 : 输入: 2736 输出: 7236 解释: 交换数字2和数字7。 示例 2 : 输入: 9973 输出: 9973 解释: 不需要交换。 注意: 给定数字的范围是 [0, 108] 模拟&#xff1a…

mysql临时表简述

概述 业务中经常会对一些表进行聚合组装信息&#xff0c;然后筛选&#xff0c;有些表比较数据量大的时候&#xff0c;会对拖慢查询&#xff1b; 常用的mybatis的分页查询&#xff0c;在查询时会先count一下所有数据&#xff0c;然后再limit分页&#xff0c;即使分页也会有深度…

数据分析-Pandas如何用图把数据展示出来

数据分析-Pandas如何用图把数据展示出来 俗话说&#xff0c;一图胜千语&#xff0c;对人类而言一串数据很难立即洞察出什么&#xff0c;但如果展示图就能一眼看出来门道。数据整理后&#xff0c;如何画图&#xff0c;画出好的图在数据分析中成为关键的一环。 数据表&#xff…

「JavaSE」抽象类接口2

&#x1f387;个人主页&#xff1a;Ice_Sugar_7 &#x1f387;所属专栏&#xff1a;快来卷Java啦 &#x1f387;欢迎点赞收藏加关注哦&#xff01; 抽象类&接口2 &#x1f349;接口间的继承&#x1f349;接口的应用&#x1f349;总结 &#x1f349;接口间的继承 和类的继承…

【每日一题】最长交替子数组

文章目录 Tag题目来源解题思路方法一&#xff1a;双层循环方法二&#xff1a;单层循环 写在最后 Tag 【双层循环】【单层循环】【数组】【2024-01-23】 题目来源 2765. 最长交替子数组 解题思路 两个方法&#xff0c;一个是双层循环&#xff0c;一个是单层循环。 方法一&am…

Structured Streaming基础--学习笔记

Structured streaming介绍 spark进行实时数据流计算时有两个工具: Spark Streaming:编写rdd代码处理数据流,可以解决非结构化的流式数据Structured Streaming:编写df代码处理数据流,可以解决结构化和半结构化的流式数据1,数据相关介绍 有界数据和无界数据 ①有界数据: …

yarn集群HDFS datanode无法启动问题排查

一、问题场景 hdfs无法访问&#xff0c;通过jps命令查看进程&#xff0c;发现namenode启动成功&#xff0c;但是所有datanode都没有启动&#xff0c;重启集群&#xff08;start-dfs.sh&#xff09;后仍然一样 二、原因分析 先看下启动的日志有无报错。打开Hadoop的日志目录 …

线程池中线程数量与队列大小参数的如何设置实践-基于QPS的计算公式

目录 概要 传统方式? 线程池理解? 基于QPS的设置思路? 总结? 概要 线程池是个既靠谱但又陌生的家伙, 像管家一样, 会踏踏实实的把你交代的任务完成, 但很死板, 没有自动安排人的能力, 需要你给它配好人手(线程实例)和承载容量(队列大小), 这些参数关系影响业务服务整体…

rabbitmq基础-java-5、Topic交换机

1、简介 Topic类型的Exchange与Direct相比&#xff0c;都是可以根据RoutingKey把消息路由到不同的队列。 只不过Topic类型Exchange可以让队列在绑定BindingKey 的时候使用通配符&#xff01; BindingKey 一般都是有一个或多个单词组成&#xff0c;多个单词之间以.分割&#x…

常用电机的分类简介

常用电机的分类简介 文章目录 常用电机的分类简介前言有刷直流电机步进电机BLDC/PMSM重点讲讲前言 电机可以将电能转换成机械能,在现代社会中广泛地应用在各种设备和系统中,家用电器,工业生产设备,交通工具,几乎随处可见。 电机通常由两个主要部分组成:定子和转子; 定…

数据库选型其实技术维度不太重要

看到这个标题可能觉得我在乱说&#xff0c;数据库选型要从多个角度和维度看来&#xff0c;还有各种POC。很多供应商朋友告诉我POC是一个漫长的过程&#xff0c;非常痛苦&#xff0c;要解决各种技术问题。怎么能说和技术无关呢&#xff1f; 因为从我的经历和周围听说的经验来说…

使用Python的pygame库实现迷宫游戏

使用Python的pygame库实现迷宫游戏 关于Python中pygame游戏模块的安装使用可见 https://blog.csdn.net/cnds123/article/details/119514520 先给出效果图&#xff1a; 这个游戏能自动生成迷宫布局。 在这个游戏中&#xff0c;玩家将使用键盘箭头键来移动&#xff0c;并且目标…

深入了解达梦数据库的增删查改操作:从入门到精通

目录 前言&#xff1a; 一.达梦数据库的增删改查 1.创建数据库 2.插入数据 3.查看数据 4.删除数据 5.数据 前言&#xff1a; 在当今数字化的时代&#xff0c;数据库已经成为企业和组织的核心资产&#xff0c;是实现高效数据处理、存储和管理的重要工具。达梦数据库&…

测试用例评审流程

1:评审的过程 A:开始前做好如下准备 1、确定需要评审的原因 2、确定进行评审的时机 3、确定参与评审人员 4、明确评审的内容 5、确定评审结束标准 6、提前至少一天将需要评审的内容以邮件的形式发送给评审会议相关人员。并注明详审时间、地点及偿参与人员等。 7、 在邮件中提醒…