【java实现+4种变体完整例子】排序算法中【桶排序】的详细解析,包含基础实现、常见变体的完整代码示例,以及各变体的对比表格

news2025/7/16 2:31:36

以下是桶排序的详细解析,包含基础实现、常见变体的完整代码示例,以及各变体的对比表格:
在这里插入图片描述


一、桶排序基础实现

原理

将数据分到有限数量的桶中,每个桶内部使用其他排序算法(如插入排序或快速排序),最后合并所有桶的结果。

代码示例
import java.util.ArrayList;
import java.util.List;

public class BucketSort {
    void sort(float[] arr) {
        if (arr.length == 0) return;

        // 创建桶(假设每个桶是一个ArrayList)
        int bucketCount = arr.length; // 桶的数量通常与元素数量相近
        List<ArrayList<Float>> buckets = new ArrayList<>();
        for (int i = 0; i < bucketCount; i++) {
            buckets.add(new ArrayList<>());
        }

        // 将数据分配到桶中
        for (float num : arr) {
            int index = (int) Math.floor(num * bucketCount);
            buckets.get(index).add(num);
        }

        // 对每个桶进行排序(此处使用插入排序)
        for (ArrayList<Float> bucket : buckets) {
            insertionSort(bucket);
        }

        // 合并所有桶的结果到原数组
        int index = 0;
        for (ArrayList<Float> bucket : buckets) {
            for (float num : bucket) {
                arr[index++] = num;
            }
        }
    }

    // 插入排序用于桶内排序
    private void insertionSort(ArrayList<Float> list) {
        for (int i = 1; i < list.size(); i++) {
            float key = list.get(i);
            int j = i - 1;
            while (j >= 0 && list.get(j) > key) {
                list.set(j + 1, list.get(j));
                j--;
            }
            list.set(j + 1, key);
        }
    }
}
复杂度分析
  • 时间复杂度
    • 平均:O(n + k)k为桶的数量)。
    • 最坏:O(n²)(数据分布极不均匀时,桶内排序退化)。
  • 空间复杂度O(n + k)
  • 稳定性:稳定(若桶内排序算法稳定)。

二、常见变体及代码示例

1. 自适应桶排序(动态桶大小)

改进点:根据数据分布动态调整桶的大小,减少极端分布的影响。
适用场景:数据分布不均匀时。

import java.util.ArrayList;
import java.util.List;

public class AdaptiveBucketSort {
    void sort(float[] arr) {
        if (arr.length == 0) return;

        // 统计数据分布
        float min = Arrays.stream(arr).min().getAsFloat();
        float max = Arrays.stream(arr).max().getAsFloat();
        int bucketCount = arr.length;
        float bucketSize = (max - min) / bucketCount;

        List<ArrayList<Float>> buckets = new ArrayList<>();
        for (int i = 0; i < bucketCount; i++) {
            buckets.add(new ArrayList<>());
        }

        // 动态分配到桶
        for (float num : arr) {
            int index = (int) ((num - min) / bucketSize);
            index = Math.min(index, bucketCount - 1); // 防止溢出
            buckets.get(index).add(num);
        }

        // 桶内排序并合并
        int index = 0;
        for (ArrayList<Float> bucket : buckets) {
            insertionSort(bucket);
            for (float num : bucket) {
                arr[index++] = num;
            }
        }
    }

    private void insertionSort(ArrayList<Float> list) {
        // 同基础版本的插入排序
    }
}
2. 分布式桶排序

改进点:利用多线程对多个桶并行排序。
适用场景:大数据或分布式计算环境。

import java.util.ArrayList;
import java.util.List;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.TimeUnit;

public class ParallelBucketSort {
    void sort(float[] arr) {
        if (arr.length == 0) return;

        int bucketCount = arr.length;
        List<ArrayList<Float>> buckets = new ArrayList<>();
        for (int i = 0; i < bucketCount; i++) {
            buckets.add(new ArrayList<>());
        }

        // 分配到桶
        for (float num : arr) {
            int index = (int) Math.floor(num * bucketCount);
            buckets.get(index).add(num);
        }

        // 并行排序每个桶
        ExecutorService executor = Executors.newFixedThreadPool(Runtime.getRuntime().availableProcessors());
        for (ArrayList<Float> bucket : buckets) {
            executor.submit(() -> insertionSort(bucket));
        }
        executor.shutdown();
        try {
            executor.awaitTermination(Long.MAX_VALUE, TimeUnit.NANOSECONDS);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }

        // 合并结果
        int index = 0;
        for (ArrayList<Float> bucket : buckets) {
            for (float num : bucket) {
                arr[index++] = num;
            }
        }
    }

    private void insertionSort(ArrayList<Float> list) {
        // 同基础版本的插入排序
    }
}
3. 基于链表的桶排序

改进点:用链表存储桶中的元素,避免动态扩容开销。
适用场景:频繁插入/删除元素的场景。

import java.util.ArrayList;
import java.util.LinkedList;
import java.util.List;

public class LinkedListBucketSort {
    void sort(float[] arr) {
        if (arr.length == 0) return;

        int bucketCount = arr.length;
        List<LinkedList<Float>> buckets = new ArrayList<>();
        for (int i = 0; i < bucketCount; i++) {
            buckets.add(new LinkedList<>());
        }

        // 分配到桶
        for (float num : arr) {
            int index = (int) Math.floor(num * bucketCount);
            buckets.get(index).add(num);
        }

        // 对每个桶排序(此处用快速排序)
        for (LinkedList<Float> bucket : buckets) {
            quickSort(bucket, 0, bucket.size() - 1);
        }

        // 合并结果
        int index = 0;
        for (LinkedList<Float> bucket : buckets) {
            for (float num : bucket) {
                arr[index++] = num;
            }
        }
    }

    private void quickSort(LinkedList<Float> list, int low, int high) {
        // 快速排序实现(略)
    }
}

三、变体对比表格

变体名称时间复杂度空间复杂度稳定性主要特点适用场景
基础桶排序O(n + k)(平均)
O(n²)(最坏)
O(n + k)稳定简单实现,适用于均匀分布数据值域均匀且数据量适中的场景
自适应桶排序O(n + k)O(n + k)稳定动态调整桶大小,适应不均匀分布数据分布不均匀但需稳定性
分布式桶排序O(n/k + k)(并行)O(n + k)稳定并行加速,适合大数据或分布式系统大数据集或高性能计算环境
基于链表的桶排序O(n + k)O(n + k)不稳定链表存储减少扩容开销,桶内排序可选算法需频繁插入/删除或对内存敏感场景

四、关键选择原则

  1. 基础场景:优先使用基础桶排序,因其简单且适合均匀分布数据。
  2. 数据分布不均匀:自适应桶排序通过动态调整桶大小,减少极端情况的影响。
  3. 性能优化:分布式版本利用多线程加速,适合大数据或并行环境。
  4. 内存效率:链表桶排序减少动态扩容开销,但需注意桶内排序算法的稳定性。
  5. 稳定性需求:若需稳定排序,避免使用链表桶排序(若桶内使用快速排序等不稳定算法)。

通过选择合适的变体,可在特定场景下优化性能或适应数据特性。例如,自适应桶排序解决数据分布问题,而分布式版本提升处理超大数据的效率。

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

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

相关文章

计算机三级:信息安全基础技术与原理(2.1密码技术简单梳理)

以下是密码学发展历程的表格归纳: ​发展阶段​时间范围​关键节点与标志性技术​技术突破与核心贡献​古典密码时期古代至19世纪• 公元前17世纪 克里特岛Phaistos圆盘(未知符号加密) • 中国西周“阴符”、北宋五言诗密码 • 1466年 艾伯蒂多表代替密码 • 1883年 克尔克霍…

【每天一个知识点】模式识别

“模式识别”是一种从数据中识别出规律、结构或趋势的技术&#xff0c;它广泛应用于人工智能、机器学习、图像处理、语音识别、自然语言处理等领域。简单来说&#xff0c;就是让计算机学会“看出”数据中的规律&#xff0c;比如&#xff1a; 从图像中识别人脸&#xff08;人脸识…

Codeforces Educational Round 177 Div. 2 【B题,C待补

B 二分 题意 样例 5 3 10 3 4 2 1 512 找最右边的L下标即可 思路 二分最靠右的L端点&#xff0c;R端点取最右端(n*k处)&#xff0c;找到后&#xff0c;答案就是L的位置(pos)&#xff0c;&#xff08;因为如果pos满足&#xff0c;则pos左边的所有下标都满足 代码 const in…

哈夫曼编码和哈夫曼树

哈夫曼编码&#xff08;Huffman Coding&#xff09; 是一种基于字符出现频率的无损数据压缩算法&#xff0c;通过构建哈夫曼树&#xff08;Huffman Tree&#xff09; 来生成最优前缀编码&#xff0c;使得高频字符用短编码&#xff0c;低频字符用长编码&#xff0c;从而实现高效…

中西面点实训室虚拟仿真操作平台

在餐饮行业蓬勃发展的当下&#xff0c;中西面点作为其中极具特色与市场需求的重要分支&#xff0c;对于专业人才的渴望愈发强烈。一个功能完备、设施先进的中西面点实训室&#xff0c;已然成为培养高素质面点专业人才的关键阵地。凯禾瑞华——实训室建设 一、中西面点实训室建设…

C++游戏服务器开发之⑦redis的使用

目录 1.当前进度 2.守护进程 3.进程监控 4.玩家姓名添加文件 5.文件删除玩家姓名 6.redis安装 7.redis存取命令 8.redis链表存取 9.redis程序结构 10.hiredisAPI使用 11.基于redis查找玩家姓名 12.MAKEFILE编写 13.游戏业务实现总结 1.当前进度 2.守护进程 3.进程监…

模拟投资大师思维:AI对冲基金开源项目详解

这里写目录标题 引言项目概述核心功能详解多样化的AI投资智能体灵活的运行模式透明的决策过程 安装和使用教程环境要求安装步骤基本使用方法运行对冲基金模式运行回测模式 应用场景和实际价值教育和研究价值潜在的商业应用与现有解决方案的对比局限性与发展方向 结论 引言 随着…

Cocos Creater打包安卓App添加隐私弹窗详细步骤+常见问题处理

最终演示效果,包含所有代码内容 + 常见错误问题处理 点击服务协议、隐私政策,跳转到相关网页, 点击同意进入游戏,不同意关闭应用 一,添加Activity,命名为MyLaunchActivity 二,编写MyLaunchActivity.java的内容 package com.cocos.game.launch;import android.os.Bund…

Android 热点二维码简单示例

Android 热点二维码简单示例 一、前言 Android 原生设置有热点二维码分享功能&#xff0c;有些系统应用也会有这个需求。 下面看看是如何实现的。 本文是一个比较简单的内容。 二、热点二维码生成实现 1、效果 整个应用就一个普通的Activity&#xff0c;显示一个按钮和二维…

JAVAEE(网络原理—UDP报头结构)

我们本篇文章要讲的是UDP的报头结构以及注意事项。 下面呢&#xff0c;我先说一下UDP是什么&#xff1f; 1.UDP是什么&#xff1f; UDP是一种网络协议。网络协议是计算机网络中&#xff0c;为了使不同设备之间能够准确、高效地进行数据交换和通信&#xff0c;而预先制定的一…

通过docker create与export来分析诊断故障镜像

&#x1f9d1; 博主简介&#xff1a;CSDN博客专家&#xff0c;历代文学网&#xff08;PC端可以访问&#xff1a;https://literature.sinhy.com/#/?__c1000&#xff0c;移动端可微信小程序搜索“历代文学”&#xff09;总架构师&#xff0c;15年工作经验&#xff0c;精通Java编…

LINUX419 更换仓库(没换成)find命令

NAT模式下虚拟机需与网卡处在同一个网段中吗 和VM1同个网段 会不会影响 这个很重要 是2 改成点2 倒是Ping通了 为啥ping百度 ping到别的地方 4399 倒是ping通了 准备下载httpd包 下不下来 正在替换为新版本仓库 报错 failure: repodata/repomd.xml from local: [Er…

鸿蒙学习笔记(5)-HTTP请求数据

一、Http请求数据 http模块是鸿蒙内置的一个模块&#xff0c;提供了网络请求的能力。不需要再写比较原始的AJAS代码。 ps:在项目中如果要访问网络资源&#xff0c;不管是图片文件还是网络请求&#xff0c;必须给项目开放权限。 &#xff08;1&#xff09;网络连接方式 HTTP数…

Spark-SQL核心编程

Spark-SQL核心编程 数据加载与保存 加载数据 spark.read.load 是加载数据的通用方法。如果读取不同格式的数据&#xff0c;可以对不同的数据格式进行设定 保存数据 df.write.save 是保存数据的通用方法。如果保存不同格式的数据&#xff0c;可以对不同的数据格式进行设定 …

LVGL源码(9):学会控件的使用(自定义弹窗)

LVGL版本&#xff1a;8.3 LVGL的控件各式各样&#xff0c;每种控件都有自己的一些特性&#xff0c;当我们想要使用一个LVGL控件时&#xff0c;我们首先可以通过官网去了解控件的一些基本特性&#xff0c;官网链接如下&#xff1a; LVGL Basics — LVGL documentation&#xf…

8、表单控制:预言水晶球——React 19 复杂表单处理

一、水晶球的预言本质 "每个表单都是时空裂缝中的预言容器&#xff0c;"占卜课教授特里劳妮凝视着水晶球&#xff0c;"React-Hook-Form与Formik的融合&#xff0c;让数据捕获如同捕捉未来碎片&#xff01;" ——以魔法部神秘事务司的预言厅为隐喻&#xf…

8 编程笔记全攻略:Markdown 语法精讲、Typora 编辑器全指南(含安装激活、基础配置、快捷键详解、使用技巧)

1 妙笔在手&#xff0c;编程无忧&#xff01; 1.1 编程为啥要做笔记&#xff1f;这答案绝了&#xff01; 嘿&#xff0c;各位键盘魔法师&#xff01;学编程不记笔记&#xff0c;就像吃火锅不配冰可乐 —— 爽到一半直接噎住&#xff01;你以为自己脑子是顶配 SSD&#xff0c;结…

Linux(autoDL云服务器)mamba-ssm环境安装——一次成功!

1.创建环境选择torch2.0&#xff0c; cuda11.8&#xff0c;python3.8 2.从GitHub官网下载cp38对应的&#xff0c;causl_conv1d&#xff0c;和mamba-ssm2.2.2。下载入下图所示。 3.直接用finalshell 或者xshell连接服务器上传&#xff0c;到根目录下面。 直接用pip install *…

代码审计入门 原生态sql注入篇

前置知识&#xff1a; 漏洞形成的原因&#xff1a; 1、可控的参数 2、函数缺陷 代码审计的步骤&#xff1a; 1、全局使用正则搜索 漏洞函数 &#xff0c;然后根据函数看变量是否可控&#xff0c;再看函数是否有过滤 2、根据web的功能点寻找函数&#xff0c;然后根据函数看…

spring Ai---向量知识库(一)

在一些垂直领域以及公司内部信息相关或者实时性相关的大模型应用&#xff0c;就无法直接使用chatGPT。 这个时候&#xff0c;向量知识库就进入了。 通过坐标向量最接近的即为匹配相关答案。 向量模型定义&#xff1a;将文档向量化&#xff0c;保证内容越相似的文本&#xff0c;…