【算法】C++用链表实现一个箱子排序附源代码详解
01 箱子排序
1.1 什么是分配排序?
分配排序的基本思想:排序过程无须比较关键字,而是通过"分配"和"收集"过程来实现排序.它们的时间复杂度可达到线性阶:O(n)。
1.2 什么是箱子排序?
箱子排序是分配排序的一种,箱子排序也称桶排序(Bucket Sort),其基本思想是:设置若干个箱子,依次扫描待排序的记录 R[0],R[1],…,R[n-1],把关键字等于 k 的记录全都装入到第 k 个箱子里(分配),然后按序号依次将各非空的箱子首尾连接起来(收集)。
比如,要将一个班的同学按分数排序,分数范围是0-100分。需设置 101 个"箱子"(R[0],R[1],…,R[100]),排序时依次将每个同学按分数放入相应的箱子里,然后依次将这些箱子首尾相接,就得到了按分数递增序排列的一个班的同学。
1.3 关于箱子个数
箱排序中,箱子的个数取决于关键字的取值范围。
若关键字的取值范围是0到m-1的整数,则必须设置 m 个箱子。因此箱排序要求关键字的类型是有限类型,否则可能要无限个箱子。
02 链表实现箱子排序
一般情况下每个箱子中存放多少个关键字相同的记录是无法预料的,故箱子的类型应设计成链表为宜。
我们现在来讲解一个简单的例子,以便来让大家更好了解这个过程。
2.1 example
下面是一个学生链表。为了更好说明问题,我们简化了学生的存储结构。每个学生节点保存一个字符,表示学生的姓名,再存一个数字,表示学生的分数。分数范围为0-5。

2.2 箱子排序的步骤
有了上面的输入链表以后。我们采用以下步骤进行箱子排序:
1) 逐个删除输入链表的节点,然后把删除的节点分配到相应的箱子中。
2) 把每个箱子中的元素收集并链接起来,使其成为一个有序链表。
比如上面的输入链表,我们要做的是:
1) 连续删除链表的首元素,并将其插入到相对应箱子的链表头部。
2) 从最后一个箱子开始,逐个删除每个箱子的元素,并将其插入一个初始为空的链表的头部。
如下图所示:

那么排序好的链表如下:

03 动手写代码
3.1 studentRecord结构体
先来看看代码:
1struct studentRecord
2{
3 int score;
4 string name;
5
6 studentRecord() {}
7 studentRecord(int theScore, string theName) :score(theScore), name(theName) {}
8
9 int operator != (const studentRecord & x) const
10 {
11 return (score != x.score);
12 }
13 operator int() const { return score; }
14};
在studentRecord这个结构体里面,我们重载了 != 这个运算符,以便用于比较等操作。还重载了int()运算符,这样一来,借助int()转换符就可以直接对学生结构体进行+-*/等操作了。
3.2 箱子排序代码
还是先看看代码吧。
1void binSort(chain<studentRecord> & theChain, int range)
2{
3 chain<studentRecord> * bin = new chain<studentRecord>[range + 1]; // 0 to range
4 int numberOfElements = theChain.size();
5
6 for (int i = 0; i < numberOfElements; i++)
7 {
8 studentRecord record = theChain.get(0);
9 theChain.erase(0);
10
11 bin[record.score].insert(0, record);
12 }
13 for (int j = range; j >= 0; j--)
14 {
15 while (!bin[j].empty())
16 {
17 studentRecord record = bin[j].get(0);
18 bin[j].erase(0);
19 theChain.insert(0, record);
20 }
21 }
22
23 delete[] bin;
24}
该函数只有两个参数,一个是学生链表。还有一个是排序范围(设置为0~range)。函数主体就是按部就班的进行上面所说的两步操作了。这里的chain链表是事先封装好的一个类。
04 完整代码
贴上一个完整的代码:
1#include <iostream>
2#include <string>
3#include <time.h>
4#include <stdlib.h>
5#include "../03_线性表_链式描述/chain.h"
6#include "../03_线性表_链式描述/chain.cpp"
7
8using std::cout;
9using std::cin;
10using std::endl;
11using std::string;
12
13struct studentRecord
14{
15 int score;
16 string name;
17
18 studentRecord() {}
19 studentRecord(int theScore, string theName) :score(theScore), name(theName) {}
20
21 int operator != (const studentRecord & x) const
22 {
23 return (score != x.score);
24 }
25 operator int() const { return score; }
26};
27
28//override out
29ostream & operator<<(ostream & out, const studentRecord & x)
30{
31 out << x.name << " " << x.score << endl;
32 return out;
33}
34
35void binSort(chain<studentRecord> & theChain, int range)
36{
37 chain<studentRecord> * bin = new chain<studentRecord>[range + 1]; // 0 to range
38 int numberOfElements = theChain.size();
39
40 for (int i = 0; i < numberOfElements; i++)
41 {
42 studentRecord record = theChain.get(0);
43 theChain.erase(0);
44
45 bin[record.score].insert(0, record);
46 }
47 for (int j = range; j >= 0; j--)
48 {
49 while (!bin[j].empty())
50 {
51 studentRecord record = bin[j].get(0);
52 bin[j].erase(0);
53 theChain.insert(0, record);
54 }
55 }
56
57 delete[] bin;
58}
59
60int main()
61{
62 srand(time(0));
63 chain<studentRecord> students;
64 studentRecord someOne;
65 for (int i = 0; i < 100; i++)
66 {
67 char Name = i % 26 + 'A';
68 someOne.name = Name;
69 someOne.score = rand() % 101;
70 students.insert(0, someOne);
71 }
72
73 binSort(students, 100);
74 cout << " ";
75 students.output(cout);
76
77 cin.get();
78
79 return 0;
80}
最后贴上一张运行效果:

欲获取代码,请关注我们的微信公众号【程序猿声】,在后台回复:listbox 。即可下载。

推荐文章:10分钟教你用Python做个打飞机小游戏超详细教程
【算法】C++用链表实现一个箱子排序附源代码详解的更多相关文章
- 【C/C++】10分钟教你用C++写一个贪吃蛇附带AI功能(附源代码详解和下载)
C++编写贪吃蛇小游戏快速入门 刚学完C++.一时兴起,就花几天时间手动做了个贪吃蛇,后来觉得不过瘾,于是又加入了AI功能.希望大家Enjoy It. 效果图示 AI模式演示 imageimage 整 ...
- 每周一个linux命令之---uptime详解
每周一个linux命令之---uptime详解 linux命令 uptime详解 引言:从今天开始,每周更新一个对程序员有用的linux命令,我真的没敢写每天一个,我怕我坚持不下去,每周一个还是可以的 ...
- linux shell 脚本攻略学习8---md5校验,sort排序,uniq命令详解
一.校验与核实 目前最为出名的校验技术是md5sum和sha1sum,它们对文件内容使用相应的算法来生成校验和. 举例: amosli@amosli-pc:~/learn$ md5sum text.t ...
- 选择排序法-java详解案例
/** * 功能:选择排序法 * 思想:第一次从R[0]-R[N-1]中选取最小值,与R[0]交换,第二次从R[1]-R[N-1]中选取最小值,与R[1]交换, * 第三次从R[2]-R[N-1]中 ...
- 一个经典的 HTTP协议详解
1引言 HTTP是一个属于应用层的面向对象的协议,由于其简捷.快速的方式,适用于分布式超媒体信息系统.它于1990年提出,经过几年的使用与发展,得到不断地完善和扩展.目前在WWW中使用的是HTTP/1 ...
- Asp.NetMVC利用LigerUI搭建一个简单的后台管理详解(函登录验证)
上一篇 Asp.Net 中Grid详解两种方法使用LigerUI加载数据库数据填充数据分页 了解了LigerUI 中Grid的基本用法 现在结合上一篇的内容做一个简单的后台管理,当然也有前台的页面 ...
- “makefile”写法详解,一步一步写一个实用的makefile,详解 sed 's,$∗\.o[ :]*,\1.o $@ : ,g' < $@.
目的:编写一个实用的makefile,能自动编译当前目录下所有.c/.cpp源文件,支持二者混合编译.并且当某个.c/.cpp..h或依赖的源文件被修改后,仅重编涉及到的源文件,未涉及的不编译. 二要 ...
- 数据结构 1 线性表详解 链表、 栈 、 队列 结合JAVA 详解
前言 其实在学习数据结构之前,我也是从来都没了解过这门课,但是随着工作的慢慢深入,之前学习的东西实在是不够用,并且太皮毛了.太浅,只是懂得一些浅层的,我知道这个东西怎么用,但是要优化.或者是解析,就不 ...
- MySQL排序函数field()详解
在日常开发过程中,排序是经常用到的,有时候有这样的需求. 比如,需要在查询结果中根据某个字段的具体值来排序.如下面例子 上面是一张个人信息 表,假如我们想按照'seiki','iris','xut'来 ...
随机推荐
- Android APK反编译就这么简单 详解
在学习Android开发的过程你,你往往会去借鉴别人的应用是怎么开发的,那些漂亮的动画和精致的布局可能会让你爱不释手,作为一个开发者,你可能会很想知道这些效果界面是怎么去实现的,这时,你便可以对改应用 ...
- linux系统软件版本升级
在安装完软件之后,在同一层目录生成一个符号链接,并把当前软件的目录映射到这个链接上,后面的操作都只通过这个链接去做,以后升级版本的时候,把最新的软件目录映射到这个链接上就可以了. 如我刚装的apach ...
- ubuntu下搭建android开发环境之超顺畅模拟器
如果说android系统的卡,像耳边蚊子让人抓狂,那么android模拟器的卡,那就像午睡时的苍蝇.大概就是一样的恶心~~ 那么,这样的问题对于开发者肯定忍无可忍,我也一样,虽然我还没有入门,但我也一 ...
- 服务器监控方案CactiEZ V10.1
CactiEZ中文版是最简单有效的Cacti中文解决方案,整合Spine,RRDTool和美化字体.集成Thold,Monitor,Syslog,Weathermap,Realtime,Errorim ...
- Java 读取jar内的文件的超简便方法
坑爹的java课程设计,偏要用jar来运行 读取.存储jar内文件的支持也好低 存储方法: 进入jar文件其实没有说的那么困难,jar文件本质是一个zip格式的压缩文件,只是把文件后缀名改了,要用Ja ...
- Docker for mac安装
Mac安装Docker docker下载地址: https://hub.docker.com/editions/community/docker-ce-desktop-mac docker for m ...
- 电脑破解wifi密码(至少连过1次的才可以)
电脑破解wifi密码(至少连过1次的才可以) 连过的wifi密码忘记了怎么办? 只要你电脑连过的都能破解. cmd输入以下内容查看电脑连接过的wifi名字. netsh wlan show profi ...
- 对JS中函数的理解
函数本质就是功能的集合 JS中函数是对象,因此,函数名实际上仅仅是一个指向函数对象的指针,不会与某个函数绑定,所以,JS中没有重载(重载就是通过传递不同类型的参数,使两个相同函数名的函数执行不同的功能 ...
- SimpleDateFormat是线程不安全的
线程不安全的SimpleDateFormat SimpleDateFormat是线程不安全的 SimpleDateFormat是Java提供的一个格式化和解析日期的工具类,日常开发中应该经常会用到,但 ...
- [GO]runtime包及gosched的使用
Gosched:让出CPU时间片 Goexit:退出当前的协程 GOMAXPROCS:设置使用最大的CPU数量(哇,牛逼了...) package main import ( "fmt&qu ...