- 浏览: 203495 次
- 性别:
- 来自: 上海
文章分类
最新评论
-
NIghtmare28:
太好用了, 谢谢
Create Local Cloudera Parcels Repo to Save Your ASS -
oyxccyj:
你好,请问下你如上的问题解决了吗?我现在也遇到同样的问题,网上 ...
Homework - HBase Shell, Java Client and MapReduce Job -
20131007:
用java描述算法?
基础数据结构和算法二:Selection sort -
ender35:
第二种实现仅能用于数组排序
计数排序(Counting Sort) -
fy616508150:
我想知道有括号参加运算怎么办
算24算法实现
文章来源:http://hxraid.iteye.com/blog/647759
原文中有基于CPP的实现,本文将CPP代码翻译为Java。原文写得很不错,复制过来做个笔记。
全依赖“比较”操作的排序算法时间复杂度的一个下界O(N*logN)。但确实存在更快的算法。这些算法并不是不用“比较”操作,也不是想办法将比较操作的次数减少到 logN。而是利用对待排数据的某些限定性假设 ,来避免绝大多数的“比较”操作。桶排序就是这样的原理。
桶排序的基本思想
假设有一组长度为N的待排关键字序列K[1....n]。首先将这个序列划分成M个的子区间(桶) 。然后基于某种映射函数 ,将待排序列的关键字k映射到第i个桶中(即桶数组B的下标 i) ,那么该关键字k就作为B[i]中的元素(每个桶B[i]都是一组大小为N/M的序列)。接着对每个桶B[i]中的所有元素进行比较排序(可以使用快排)。然后依次枚举输出B[0]....B[M]中的全部内容即是一个有序序列。
[桶—关键字]映射函数
bindex=f(key) 其中,bindex 为桶数组B的下标(即第bindex个桶), k为待排序列的关键字。桶排序之所以能够高效,其关键在于这个映射函数,它必须做到:如果关键字k1<k2,那么f(k1)<=f(k2)。也就是说B(i)中的最小数据都要大于B(i-1)中最大数据。很显然,映射函数的确定与数据本身的特点有很大的关系,我们下面举个例子:
假如待排序列K= {49、 38 、 35、 97 、 76、 73 、 27、 49 }。这些数据全部在1—100之间。因此我们定制10个桶,然后确定映射函数f(k)=k/10。则第一个关键字49将定位到第4个桶中(49/10=4)。依次将所有关键字全部堆入桶中,并在每个非空的桶中进行快速排序后得到如下图所示:
对上图只要顺序输出每个B[i]中的数据就可以得到有序序列了。
桶排序代价分析
桶排序利用函数的映射关系,减少了几乎所有的比较工作。实际上,桶排序的f(k)值的计算,其作用就相当于快排中划分,已经把大量数据分割成了基本有序的数据块(桶)。然后只需要对桶中的少量数据做先进的比较排序即可。
对N个关键字进行桶排序的时间复杂度分为两个部分:
(1) 循环计算每个关键字的桶映射函数,这个时间复杂度是O(N)。
(2) 利用先进的比较排序算法对每个桶内的所有数据进行排序,其时间复杂度为 ∑ O(Ni*logNi) 。其中Ni 为第i个桶的数据量。
很显然,第(2)部分是桶排序性能好坏的决定因素。尽量减少桶内数据的数量是提高效率的唯一办法(因为基于比较排序的最好平均时间复杂度只能达到O(N*logN)了)。因此,我们需要尽量做到下面两点:
(1) 映射函数f(k)能够将N个数据平均的分配到M个桶中,这样每个桶就有[N/M]个数据量。
(2) 尽量的增大桶的数量。极限情况下每个桶只能得到一个数据,这样就完全避开了桶内数据的“比较”排序操作。 当然,做到这一点很不容易,数据量巨大的情况下,f(k)函数会使得桶集合的数量巨大,空间浪费严重。这就是一个时间代价和空间代价的权衡问题了。
对于N个待排数据,M个桶,平均每个桶[N/M]个数据的桶排序平均时间复杂度为:
O(N)+O(M*(N/M)*log(N/M))=O(N+N*(logN-logM))=O(N+N*logN-N*logM)
当N=M时,即极限情况下每个桶只有一个数据时。桶排序的最好效率能够达到O(N)。
总结: 桶排序的平均时间复杂度为线性的O(N+C),其中C=N*(logN-logM)。如果相对于同样的N,桶数量M越大,其效率越高,最好的时间复杂度达到O(N)。 当然桶排序的空间复杂度 为O(N+M),如果输入数据非常庞大,而桶的数量也非常多,则空间代价无疑是昂贵的。此外,桶排序是稳定的。
桶排序在海量数据中的应用
一年的全国高考考生人数为500 万,分数使用标准分,最低100 ,最高900 ,没有小数,你把这500 万元素的数组排个序。
分析:对500W数据排序,如果基于比较的先进排序,平均比较次数为O(5000000*log5000000)≈1.112亿。但是我们发现,这些数据都有特殊的条件: 100=<score<=900。那么我们就可以考虑桶排序这样一个“投机取巧”的办法、让其在毫秒级别就完成500万排序。
方法:创建801(900-100)个桶。将每个考生的分数丢进f(score)=score-100的桶中。这个过程从头到尾遍历一遍数据只需要500W次。然后根据桶号大小依次将桶中数值输出,即可以得到一个有序的序列。而且可以很容易的得到100分有***人,501分有***人。
实际上,桶排序对数据的条件有特殊要求,如果上面的分数不是从100-900,而是从0-2亿,那么分配2亿个桶显然是不可能的。所以桶排序有其局限性,适合元素值集合并不大的情况。
import java.util.Random; public class BucketSort { public static void main(String... args) { //int raw[]={49, 38, 65, 97, 76, 13, 27, 49}; if (args.length < 1) { System.out.println("usage: java BucketSort number"); return; } Random rand = new Random(); int count = Integer.parseInt(args[0]); int[] raw = new int[count]; for (int i = 0; i < count; i++) { raw[i] = rand.nextInt(100); } BucketSort bs = new BucketSort(); bs.bucketSort(raw, raw.length, 10); } public void bucketSort(int[] keys, int size, int bucketSize) { Node[] bucketTable = new Node[bucketSize]; for (int i = 0; i < bucketSize; i++) { bucketTable[i] = new Node(0, null); } for (int j = 0; j < size; j++) { Node node = new Node(keys[j], null); // mapping function int index = keys[j] / 10; Node n = bucketTable[index]; if (n.getKey() == 0) { bucketTable[index].setNext(node); bucketTable[index].setKey(bucketTable[index].getKey() + 1); } else { while (n.getNext() != null && n.getNext().getKey() <= node.getKey()) { n = n.getNext(); } node.setNext(n.getNext()); n.setNext(node); bucketTable[index].setKey(bucketTable[index].getKey() + 1); } } for (int b = 0; b < bucketSize; b++) { for (Node n = bucketTable[b].getNext(); n != null; n = n.getNext()) { System.out.print(n.toString()); } } System.out.println(); } } class Node { private int key; private Node next; public Node(int key, Node next) { this.key = key; this.next = next; } public int getKey() { return this.key; } public void setKey(int key) { this.key = key; } public Node getNext() { return this.next; } public void setNext(Node next) { this.next = next; } public String toString() { return this.key + "\t"; } }
end.
发表评论
-
基础数据结构和算法十四:Directed Graphs
2013-12-15 22:40 1814In directed graphs, edges a ... -
基础数据结构和算法十三:Undirected Graphs (2)
2013-12-13 22:51 1069Design pattern for graph p ... -
基础数据结构和算法十三:Undirected Graphs
2013-12-13 20:15 1210A graph is a set of vertices ... -
基础数据结构和算法十二:Hash table
2013-12-02 22:06 1059Search algorithms that use ... -
基础数据结构和算法十一:Red-black binary search tree
2013-12-01 12:12 1524The insertion algorithm fo ... -
基础数据结构和算法十:2-3 search tree
2013-11-30 11:07 1225Binary search tree works w ... -
基础数据结构和算法九:Binary Search Tree
2013-11-28 22:39 1682A binary search tree (BST) ... -
基础数据结构和算法八:Binary search
2013-11-28 21:21 1158Binary search needs an ordere ... -
基础数据结构和算法七:Priority queue & Heap sort
2013-11-27 19:47 2721Some important applications o ... -
基础算法七: Priority queue & Heap sort
2013-11-26 21:46 0Some important applications ... -
基础数据结构和算法六:Quick sort
2013-11-21 19:33 1247Quick sort is probably used m ... -
Inversions of an array
2013-11-20 22:28 0TODO -
基础数据结构和算法五:Merge sort
2013-11-20 21:44 2000One of mergesort’s most at ... -
基础数据结构和算法四:Shell sort
2013-11-20 19:11 1195Shellsort is a simple exte ... -
Comparing two sorting algorithms
2013-11-19 21:16 871Generally we compare algorith ... -
基础数据结构和算法三:Insertion Sort
2013-11-19 21:06 1012As in selection sort, the ite ... -
基础数据结构和算法二:Selection sort
2013-11-19 20:57 1185One of the simplest sortin ... -
基础数据结构和算法一:UnionFind
2013-11-19 20:47 1288The problem that we consid ... -
Three-sum with linear time complexity
2013-09-14 10:08 0TODO package org.george.algor ... -
Blooming Filter in Hadoop
2013-07-27 22:49 0TODO
相关推荐
多趟桶式排序bucket sort。数据输入:有10个数,范围为0到999。 [root@lumotuwe] gcc cas.c -o cas [root@lumotuwe] ./cas 64 8 216 512 27 729 0 1 343 125
提供的`bucketSort.cc`文件可能包含一个桶排序的C++实现,通过阅读源代码,你可以深入理解桶排序的具体步骤和细节,包括映射函数的设计、桶的创建和数据的分布等。 桶排序是一种非常实用的排序算法,尤其在处理大...
经典排序算法 - 桶排序Bucket sort 经典排序算法 - 插入排序Insertion sort 经典排序算法 - 基数排序Radix sort 经典排序算法 - 鸽巢排序Pigeonhole sort 经典排序算法 - 归并排序Merge sort 经典排序算法 - ...
桶排序算法是一种基于计数的排序算法,其基本思想是先扫描一遍待排序的序列,找出最大值和最小值,然后根据最大值和最小值确定桶的个数,并将每个元素分配到对应的桶中,最后将每个桶中的元素进行排序,然后依次输出...
MPI bucket排序代码 用于mpi联系 熟悉bucket排序
在提供的文件名列表中,"bucketsort.c"和"BucketSort.cpp"可能是实现MPI桶排序的具体代码。其中,"bucketsort.c"可能使用C语言编写,而"BucketSort.cpp"则可能采用C++语言。这两个文件通常会包含以下部分: 1. **...
在本主题中,我们将深入探讨一种特殊的排序算法——**BucketSort(桶排序)**,并将其与C语言的实现相结合。 **BucketSort**是一种分布式排序算法,其核心思想是将待排序的数据分布到若干个有序的“桶”中,每个桶...
C++桶排序是一种非比较型排序算法,它的工作原理是将一组元素分布到有限数量的“桶”里,每个桶再个别排序,最后按顺序连接所有桶得到有序序列。这种算法特别适合排序范围很广的集合。桶排序常用于将数据均匀分布在...
BucketSort为AlgorithmMan中的桶排序演示工具(这是可执行文件;需要.net 4.0支持;非源代码)。 原文:C#算法设计排序篇之10-桶排序(附带动画演示程序) 链接:...
桶排序(Bucket Sort)是一种分布式排序算法,它将要排序的数据分布到多个“桶”中,每个桶再分别进行排序,最后将所有桶中的数据合并,得到全局有序的结果。在iOS开发中,桶排序可以被用于优化大规模数据的排序过程...
桶排序(Bucket Sort)是一种分布式排序算法,它将待排序的数据分布到多个“桶”中,每个桶再分别进行排序,最后将所有桶中的数据合并,得到全局有序的结果。这个算法通常适用于数据分布均匀的情况,如果数据在一定...
C++实现的BucketSort涉及到的关键概念包括桶的创建、数据分布、桶内排序和结果合并。在实际编程时,需要灵活地调整桶的数量和大小,以适应不同类型的输入数据,同时选择合适的桶内排序算法,以达到最佳的排序效果。...
桶排序(Bucket Sort)是一种非比较型的排序算法,它基于分布排序的原理,通过将待排序的数据分到几个有序的桶里,每个桶再分别排序,最后将所有桶中的数据合并成一个有序序列。这种算法特别适用于数据分布均匀的...
"7 桶排序bucketsort.swf"动画解释了如何根据数据分布情况,将元素分到不同的桶中,然后对每个桶内的元素进行排序。 八、希尔排序(Shell Sort) 希尔排序是插入排序的一种更高效的改进版本,它通过将相距一定间隔...
例如,提供的`BucketSort.docx`文件可能包含了桶排序的详细步骤、代码示例以及不同情况下的性能分析。通常,这样的文档会讲解如何创建和管理桶,如何选择合适的排序算法对桶内元素排序,以及如何优化桶排序的实现以...
桶排序(Bucket Sort)是一种分布式排序算法,它将要排序的数据分到几个有序的桶里,每个桶再分别排序,最后按照每个桶内元素的顺序依次取出,组合成整体的有序序列。这种算法适用于数据分布相对均匀的情况。下面...
桶排序(Bucket Sort)是一种分布式排序算法,它将要排序的数据分布到多个“桶”中,每个桶再分别进行排序,最后将所有桶中的数据合并,得到全局有序的结果。桶排序算法的时间复杂度在最理想情况下可以达到线性的O(n...
随机产生一定范围内的随机数,进行桶排序,然后二分查找任意个数的数。
桶排序(Bucket Sort)是一种分布式排序算法,常用于大数据处理和流式计算中。它将待排序的数据分布到多个“桶”中,每个桶再独立地进行排序,最后按照桶的顺序依次取出桶中的元素,形成有序序列。桶排序的核心思想...