首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >在c++中编写存储桶排序

在c++中编写存储桶排序
EN

Stack Overflow用户
提问于 2012-02-17 02:50:33
回答 3查看 20.4K关注 0票数 6

我有一本书是这样说的:

a)根据值的个位数,将一维数组的每个值放入桶数组的一行中。例如,97放在第7行,3放在第3行,100放在第0行。这就是所谓的“分发过程”。

b)逐行循环存储桶数组,并将值复制回原始数组。这被称为“聚集通行证”。一维数组中前面的值的新顺序是100、3和97。

c)对每个后续数字位置重复此过程。

我在理解和实现这一点上遇到了很多困难。到目前为止,我有:

代码语言:javascript
复制
void b_sort(int sarray[], int array_size) {
    const int max = array_size;
    for(int i = 0; i < max; ++i)
        int array[i] = sarray[i];

    int bucket[10][max - 1];
}

我在想,为了按1、10、数百等对它们进行排序,我可以使用以下代码:

代码语言:javascript
复制
for(int i = 0; i < max; ++i)
    insert = (array[i] / x) % 10;
    bucket[insert];

其中x= 1,10,100,1000,等等。我现在完全不知道该怎么写了。

EN

回答 3

Stack Overflow用户

回答已采纳

发布于 2012-02-17 03:48:22

这是一个基于OP问题中的信息的存储桶排序。

代码语言:javascript
复制
void b_sort(int sarray[], int array_size) {
    const int max = array_size;
    // use bucket[x][max] to hold the current count
    int bucket[10][max+1];
    // init bucket counters
    for(var x=0;x<10;x++) bucket[x][max] = 0;
    // main loop for each digit position
    for(int digit = 1; digit <= 1000000000; digit *= 10) {
        // array to bucket
        for(int i = 0; i < max; i++) {
            // get the digit 0-9
            int dig = (sarray[i] / digit) % 10;
            // add to bucket and increment count
            bucket[dig][bucket[dig][max]++] = sarray[i];
        }
        // bucket to array
        int idx = 0;
        for(var x = 0; x < 10; x++) {
            for(var y = 0; y < bucket[x][max]; y++) {
                sarray[idx++] = bucket[x][y];
            }
            // reset the internal bucket counters
            bucket[x][max] = 0;
        }
    }
}

注意到使用二维数组存储桶浪费了大量空间……队列/列表的数组通常更有意义。

我通常不使用C++编程,并且上面的代码是在web浏览器中编写的,因此可能存在语法错误。

票数 4
EN

Stack Overflow用户

发布于 2012-02-17 18:28:44

下面的代码使用十六进制数字进行存储桶排序(对于BITS_PER_BUCKET=4)。当然,它是有教育意义的,而不是富有成效的。

代码语言:javascript
复制
#include <assert.h>
#include <stdio.h>

#define TEST_COUNT 100
#define BITS_PER_BUCKET 4
#define BUCKET_COUNT (1 << BITS_PER_BUCKET)
#define BUCKET_MASK (BUCKET_COUNT-1)
#define PASS_COUNT (8*sizeof(int)/BITS_PER_BUCKET)

int main(int argc, char** argv) {

  printf("Starting up ...");
  assert((PASS_COUNT*BITS_PER_BUCKET) == (8*sizeof(int)));
  printf("... OK\n");

  printf("Creating repeatable very-pseudo random test data ...");
  int data[TEST_COUNT];
  int x=13;
  int i;
  for (i=0;i<TEST_COUNT;i++) {
    x=(x*x+i*i) % (2*x+i);
    data[i]=x;
  }
  printf("... OK\nData is ");
  for (i=0;i<TEST_COUNT;i++) printf("%02x, ",data[i]);
  printf("\n");

  printf("Creating bucket arrays ...");
  int buckets[BUCKET_COUNT][TEST_COUNT];
  int bucketlevel[BUCKET_COUNT];
  for (i=0;i<BUCKET_COUNT;i++) bucketlevel[i]=0;
  printf("... OK\n");

  for (i=0;i<PASS_COUNT;i++) {

    int j,k,l;

    printf("Running distribution pass #%d/%d ...",i,PASS_COUNT);
    l=0;
    for (j=0;j<TEST_COUNT;j++) {
      k=(data[j]>>(BITS_PER_BUCKET*i)) & BUCKET_MASK;
      buckets[k][bucketlevel[k]++]=data[j];
      l|=k;
    }
    printf("... OK\n");

    if (!l) {
      printf("Only zero digits found, sort completed early\n");
      break;
    }

    printf("Running gathering pass #%d/%d ...",i,PASS_COUNT);
    l=0;
    for (j=0;j<BUCKET_COUNT;j++) {
      for (k=0;k<bucketlevel[j];k++) {
        data[l++]=buckets[j][k];
      }
      bucketlevel[j]=0;
    }
    printf("... OK\nData is ");
    for (l=0;l<TEST_COUNT;l++) printf("%02x, ",data[l]);
    printf("\n");

  }
}
票数 1
EN

Stack Overflow用户

发布于 2016-11-24 08:28:51

使用STL队列在C++11中重写了路易斯的代码。

代码语言:javascript
复制
void bucket_sort(vector<int>& arr){
    queue<int> buckets[10];
    for(int digit = 1; digit <= 1e9; digit *= 10){
        for(int elem : arr){
            buckets[(elem/digit)%10].push(elem);
        }
        int idx = 0;
        for(queue<int>& bucket : buckets){
            while(!bucket.empty()){
                arr[idx++] = bucket.front();
                bucket.pop();
            }
        }
    } 
}
票数 1
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/9317248

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档