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结构体

先来看看代码:

  1. 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 箱子排序代码

还是先看看代码吧。

  1. 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. 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做个打飞机小游戏超详细教程

推荐文章:10分钟教你用python下载和拼接微信好友头像图片

推荐文章:10分钟教你用python一行代码搞点大新闻

推荐文章:10分钟教你用python打造贪吃蛇超详细教程

【算法】C++用链表实现一个箱子排序附源代码详解的更多相关文章

  1. 【C/C++】10分钟教你用C++写一个贪吃蛇附带AI功能(附源代码详解和下载)

    C++编写贪吃蛇小游戏快速入门 刚学完C++.一时兴起,就花几天时间手动做了个贪吃蛇,后来觉得不过瘾,于是又加入了AI功能.希望大家Enjoy It. 效果图示 AI模式演示 imageimage 整 ...

  2. 每周一个linux命令之---uptime详解

    每周一个linux命令之---uptime详解 linux命令 uptime详解 引言:从今天开始,每周更新一个对程序员有用的linux命令,我真的没敢写每天一个,我怕我坚持不下去,每周一个还是可以的 ...

  3. linux shell 脚本攻略学习8---md5校验,sort排序,uniq命令详解

    一.校验与核实 目前最为出名的校验技术是md5sum和sha1sum,它们对文件内容使用相应的算法来生成校验和. 举例: amosli@amosli-pc:~/learn$ md5sum text.t ...

  4. 选择排序法-java详解案例

    /**  * 功能:选择排序法 * 思想:第一次从R[0]-R[N-1]中选取最小值,与R[0]交换,第二次从R[1]-R[N-1]中选取最小值,与R[1]交换, * 第三次从R[2]-R[N-1]中 ...

  5. 一个经典的 HTTP协议详解

    1引言 HTTP是一个属于应用层的面向对象的协议,由于其简捷.快速的方式,适用于分布式超媒体信息系统.它于1990年提出,经过几年的使用与发展,得到不断地完善和扩展.目前在WWW中使用的是HTTP/1 ...

  6. Asp.NetMVC利用LigerUI搭建一个简单的后台管理详解(函登录验证)

    上一篇 Asp.Net 中Grid详解两种方法使用LigerUI加载数据库数据填充数据分页  了解了LigerUI 中Grid的基本用法  现在结合上一篇的内容做一个简单的后台管理,当然也有前台的页面 ...

  7. “makefile”写法详解,一步一步写一个实用的makefile,详解 sed 's,$∗\.o[ :]*,\1.o $@ : ,g' < $@.

    目的:编写一个实用的makefile,能自动编译当前目录下所有.c/.cpp源文件,支持二者混合编译.并且当某个.c/.cpp..h或依赖的源文件被修改后,仅重编涉及到的源文件,未涉及的不编译. 二要 ...

  8. 数据结构 1 线性表详解 链表、 栈 、 队列 结合JAVA 详解

    前言 其实在学习数据结构之前,我也是从来都没了解过这门课,但是随着工作的慢慢深入,之前学习的东西实在是不够用,并且太皮毛了.太浅,只是懂得一些浅层的,我知道这个东西怎么用,但是要优化.或者是解析,就不 ...

  9. MySQL排序函数field()详解

    在日常开发过程中,排序是经常用到的,有时候有这样的需求. 比如,需要在查询结果中根据某个字段的具体值来排序.如下面例子 上面是一张个人信息 表,假如我们想按照'seiki','iris','xut'来 ...

随机推荐

  1. Opencv 分水岭分割图片

    #include <iostream>#include <opencv2/opencv.hpp> using namespace std;using namespace cv; ...

  2. RTX二次开发::检索 COM 类工厂中 CLSID 为 {79210E58-99EB-45D0-8890-763EFEAA143F} 的组件失败,

    解决方法: 1.Interop.RTXSAPILib.dll这个是32位的 将IIS 应用程序池 启用32位应用程序 设置为true就可以了, 2.把编译好的程序选择X86平台运行,就好了 程序员的基 ...

  3. radio后的input框数据传递

    <input type="radio" name="limit_type" value="total">活动期间,每个手机号可抽 ...

  4. Maven常用配置

    运行mvn install时跳过Test 方法一: <project> [...] <build> <plugins> <plugin> <gro ...

  5. Ubuntu 将应用程序 固定到快快速启动栏(以Sublime为例)

    因为Sublime Text并不是需要安装,所以缺少Ubuntu桌面运行的一些基本配置,比如不能将它加入桌面侧边的启动器. 而Ubuntu上也没有快捷方式的说法,而通过软件中心安装的软件就有图标,并能 ...

  6. Android性能分析Systrace工具

    一.概述 保证系统流畅度,也就是保证系统能连续不间断地提供每秒60帧的运行状态.当出现掉帧时(也可称为Jank),需要知道当前整个系统所处的状态,systrace便是最佳的选择,它能手机检测Andro ...

  7. mysql中四种存储引擎的区别和选择

    前言 数据库存储引擎是数据库底层软件组织,数据库管理系统(DBMS)使用数据引擎进行创建.查询.更新和删除数据.不同的存储引擎提供不同的存储机制.索引技巧.锁定水平等功能,使用不同的存储引擎,还可以 ...

  8. Java设置jre通过java new Date()得到的时间的时区

    1.前提 由于公司有印尼的项目,该项目仅对印尼当地开放使用(公司在国内,用的是阿里云的ECS,但是阿里云在印尼没有服务器,所以就买了新加坡的服务器),印尼当地人用的是东七区的时间,所以比国内东八区的时 ...

  9. CentOS 下 zookeeper 安装

     搭建zookeeper需要几个条件 a. 配置Java环境  c. centos d. 下载 xshell5 (下载它只是为了更方便的使用linux)     一.新建一个myapp目录: 二.下载 ...

  10. [python] [转]如何自动生成和安装requirements.txt依赖

    [转]如何自动生成和安装requirements.txt依赖 在查看别人的Python项目时,经常会看到一个requirements.txt文件,里面记录了当前程序的所有依赖包及其精确版本号.这个文件 ...