0

0

C++怎么实现基数排序_C++非比较型整数排序【效率】

裘德小鎮的故事

裘德小鎮的故事

发布时间:2026-02-25 10:53:02

|

410人浏览过

|

来源于php中文网

原创

std::sort不是基数排序的替代品,因其为比较型排序、最坏o(n log n),而基数排序对固定位宽整数可达o(n×k);但需手动控制位宽、进制与内存布局,标准库不提供接口。

c++怎么实现基数排序_c++非比较型整数排序【效率】

为什么 std::sort 不是基数排序的替代品

因为 std::sort 是比较型排序,最坏 O(n log n),而基数排序对整数可做到 O(n × k)(k 是位数),当数据范围固定、n 很大时,实际更快。但它的前提是:你得自己控制位宽、进制和内存布局——标准库不提供现成接口。

常见错误现象:std::sort 在处理上百万个 int32_t 时比手写基数排序慢 1.5–2 倍;有人试图用 std::stable_sort 模拟计数排序,结果因内存分配抖动反而更慢。

  • 适用场景:批量排序无符号整数(uint32_tsize_t)、或已归一化为非负整数的 key(比如哈希桶索引、离散化后的 ID)
  • 不适用:带符号整数直接丢进去会错——负数补码最高位为 1,排在正数后面,需先做偏移转换
  • 性能关键点:每轮计数排序的桶数组必须栈分配(如 int bucket[256]),堆分配或 std::vector 会显著拖慢

如何用 8-bit 分桶实现 uint32_t 的 4 轮基数排序

按字节拆分,每轮处理 8 位,共 4 轮。比 16-bit 分桶更省内存(256 个桶 vs 65536),缓存友好,实测在 x86-64 上吞吐更高。

实操建议:

立即学习C++免费学习笔记(深入)”;

超级简历WonderCV
超级简历WonderCV

免费求职简历模版下载制作,应届生职场人必备简历制作神器

下载
  • 输入数组必须是 std::vector<uint32_t></uint32_t> 或裸指针 + size,避免迭代器抽象开销
  • 每轮用两个缓冲区交替:原数组 → 临时数组 → 原数组,避免拷贝全部数据,只交换指针/引用
  • 计数阶段用 bucket[i] = 0 初始化,别用 std::fill ——循环展开或 memset 更快
  • 偏移累加时用前缀和:遍历 bucket,把 bucket[i] 改成 “小于等于 i 的元素总数”,方便后续直接定位写入位置

示例关键片段(第 0 轮,处理最低字节):

for (size_t i = 0; i < n; ++i) {
    uint8_t b = static_cast<uint8_t>(arr[i]);
    bucket[b]++;
}
// 前缀和
for (int i = 1; i < 256; ++i) {
    bucket[i] += bucket[i-1];
}
// 逆序写入(保证稳定)
for (size_t i = n; i-- > 0; ) {
    uint8_t b = static_cast<uint8_t>(arr[i]);
    out[--bucket[b]] = arr[i];
}

处理 int32_t 时怎么避免符号位搞乱顺序

补码下 -1 == 0xFFFFFFFF,比 INT_MAX 还大,直接排序会让负数全排在最后。不能靠 abs(),会溢出;也不能简单加偏移(如 +2147483648),得用位操作绕过符号解释。

正确做法:把 int32_t 当作 uint32_t 重解释,再异或一个掩码翻转符号位。

  • 转换公式:uint32_t key = *p ^ 0x80000000Up 指向 int32_t
  • 原理:把最高位从“符号位”变成“大小位”——原来负数高位为 1,异或后变 0,自然排在前面;正数高位为 0,变 1,排在后面
  • 还原时再异或一次即可,无需额外存储;整个过程零开销,编译器通常优化成单条 xor 指令
  • 错误示范:用 static_cast<uint32_t>(x) + 0x80000000U</uint32_t> —— 对负数是未定义行为(有符号转无符号在 C++20 前依赖实现)

什么时候该放弃基数排序改回 std::sort

基数排序不是银弹。当数据量小(n )、或 key 分布极度稀疏(比如全是 <code>00x7FFFFFFF)、或内存受限时,它反而更慢甚至崩掉。

  • 缓存失效:每轮扫描整个数组 + 随机写入 256 个桶的起始位置,若 n 太小,CPU 缓存反复换入换出,不如 std::sort 的局部访问模式
  • 分支预测失败:桶计数循环中 bucket[b]++b 若高度随机,现代 CPU 的分支预测器会大量失准
  • 容易被忽略的点:如果你要排序的是结构体数组(比如 struct { int key; float val; }),按 key 排序时必须同时搬运整个 struct —— 此时每轮 memcpy 成本飙升,除非你用索引间接排序,否则收益可能归零

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

智谱清言 - 免费全能的AI助手
智谱清言 - 免费全能的AI助手

智谱清言 - 免费全能的AI助手

相关专题

更多
css中float用法
css中float用法

css中float属性允许元素脱离文档流并沿其父元素边缘排列,用于创建并排列、对齐文本图像、浮动菜单边栏和重叠元素。想了解更多float的相关内容,可以阅读本专题下面的文章。

592

2024.04.28

C++中int、float和double的区别
C++中int、float和double的区别

本专题整合了c++中int和double的区别,阅读专题下面的文章了解更多详细内容。

105

2025.10.23

sort排序函数用法
sort排序函数用法

sort排序函数的用法:1、对列表进行排序,默认情况下,sort函数按升序排序,因此最终输出的结果是按从小到大的顺序排列的;2、对元组进行排序,默认情况下,sort函数按元素的大小进行排序,因此最终输出的结果是按从小到大的顺序排列的;3、对字典进行排序,由于字典是无序的,因此排序后的结果仍然是原来的字典,使用一个lambda表达式作为key参数的值,用于指定排序的依据。

404

2023.09.04

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

365

2025.06.09

golang结构体方法
golang结构体方法

本专题整合了golang结构体相关内容,请阅读专题下面的文章了解更多。

200

2025.07.04

string转int
string转int

在编程中,我们经常会遇到需要将字符串(str)转换为整数(int)的情况。这可能是因为我们需要对字符串进行数值计算,或者需要将用户输入的字符串转换为整数进行处理。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

850

2023.08.02

int占多少字节
int占多少字节

int占4个字节,意味着一个int变量可以存储范围在-2,147,483,648到2,147,483,647之间的整数值,在某些情况下也可能是2个字节或8个字节,int是一种常用的数据类型,用于表示整数,需要根据具体情况选择合适的数据类型,以确保程序的正确性和性能。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

584

2024.08.29

c++怎么把double转成int
c++怎么把double转成int

本专题整合了 c++ double相关教程,阅读专题下面的文章了解更多详细内容。

294

2025.08.29

batoto漫画官网入口与网页版访问指南
batoto漫画官网入口与网页版访问指南

本专题系统整理batoto漫画官方网站最新可用入口,涵盖最新官网地址、网页版登录页面及防走失访问方式说明,帮助用户快速找到batoto漫画官方平台,稳定在线阅读各类漫画内容。

15

2026.02.25

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
C# 教程
C# 教程

共94课时 | 10.1万人学习

C 教程
C 教程

共75课时 | 4.9万人学习

C++教程
C++教程

共115课时 | 19.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号