【八大排序之交换与归并排序】(一)

简介: 【八大排序之交换与归并排序】(一)

1 交换排序

1.1 冒泡排序

基本思想:

  • 所谓交换,就是根据序列中两个记录键值的比较结果来对换这两个记录在序列中的位置,交换排序的特点是:将键值较大的记录向序列的尾部移动,键值较小的记录向序列的前部移动(升序)。
  • 冒泡排序的图解:
  • 5dda632d96124b90a5cd7ffcb66ec2cc.png
  • 冒泡排序的特性总结:
  • 1. 冒泡排序是一种非常容易理解的排序
  • 2. 时间复杂度:O(N^2)
  • 3. 空间复杂度:O(1)
  • 4. 稳定性:稳定 (可以控制相同数据的相对位置不发生改变)
  • 具体代码:
void Swap(int* p1, int* p2)
{
  int tmp = *p1;
  *p1 = *p2;
  *p2 = tmp;
}
void BubbleSort(int* a, int sz)
{
  int i = 0;
  int j = 0;
    int flag=0;
  for (i = 0; i < sz - 1; i++)
  {
    for (j = 0; j < sz - 1 - i; j++)
    {
      if (a[j] > a[j + 1])
      {
        Swap(&a[j], &a[j + 1]);
                flag=1;
      }
    }
        if(0==flag)
        {
            break;
        }
  }
}

冒泡排序比较简单,写起来也很容易,这里就不再多说了。

1.2 快速排序(重点)

基本思想:

快速排序是 Hoare 于 1962 年提出的一种二叉树结构的交换排序方法,其基本思想为:任取待排序元素序列中的某元素作为基准值,按照该排序码将待排序集合分割成两子序列, 左子序列中所有元素均小于基准值,右子序列中所有元素均大于基准值 ,然后最左右子序列重复该过程,直到所 有元素都排列在相应位置上为止。

将区间按照基准值划分为左右两半部分的常见方式有:

a 挖坑法

b 前后指针法

c 左右指针法(Hoare法)

1.2.1 挖坑法

基本思想:

首先寻找一个坑位(一般选最左边或者中间或者最右边)假设我这里选最左边为坑位,然后用一个关键值key保存坑位值;再从最右边找到小于key的值,找到后就将坑位的值改为找到后的值,然后将找到位置的下标作为新的坑位;再从左边去找到大于坑位的值,找到后同样将坑位的值改为找到后的值,然后将找到位置的下标作为新的坑位;直至相遇。最后将相遇点的值改为key.  这样就完成了将key排到了正确位置。然后分治,用递归将左边与右边分别排,直至所有元素都排到了正确位置。

具体代码实现:

void Swap(int* p1, int* p2)
{
  int tmp = *p1;
  *p1 = *p2;
  *p2 = tmp;
}
void QuickSort(int* a, int left, int right)
{
  if (left >= right)
  {
    return;
  }
  int begin = left;
  int end = right;
  int pivot = begin;
  int key = a[pivot];
  while (begin < end)
  {
    //右边找小 放在左边
    while (begin < end && a[end] >= key)
    {
      end--;
    }
    a[pivot] = a[end];
    pivot = end;
    //左边找大 放在右边
    while (begin < end && a[begin] <= key)
    {
      begin++;
    }
    a[pivot] = a[begin];
    pivot = begin;
  }
  a[pivot] = key;
  QuickSort(a, left, pivot - 1);
  QuickSort(a, pivot+1, right);
}

注意事项:

  1. 进行key值单趟排序也可以单独分装一个PartSort函数来实现(通过返回值来确定已经排序好的key值),这里我就不分装了,后面前后指针与左右指针采用的是分装过的。
  2. 右边找小和左边找大时要加上=,否则可能会造成死循环。

优化版本:

优化版本主要从两点优化:

1 实现快排的3中方法比较高效的原因就是用了二叉树的思想先排出较靠近中间值的元素,但是上面的方法并不能保证我们选择的key是较为靠近中间值,所以我们用一个3数取中的方法来处理。

2 由于排最后的几个元素用快排的效率较低,所以我们用直接插入来排(小区间优化)

具体代码:

//3数取中
int GetMidIndex(int* a, int left, int right)
{
  int mid = (left + right) / 2;
  if (a[left] < a[mid])
  {
    if (a[right] > a[mid])
      return mid;
    else if (a[right] < a[left])
      return left;
    else
      return right;
  }
  else
  {
    if (a[right] > a[left])
      return left;
    else if (a[right] < a[mid])
      return mid;
    else
      return right;
  }
}
void QuickSort(int* a, int left, int right)
{
  if (left >= right)
  {
    return;
  }
  int index = GetMidIndex(a, left, right);
  Swap(&a[left], &a[index]);
  int start = left;
  int end = right;
  int pivot = start;
  int key = a[pivot];
  while (start < end)
  {
    while (start < end && a[end] <= key)
    {
      end--;
    }
    a[pivot] = a[end];
    pivot = end;
    while (start < end && a[start] >= key)
    {
      start++;
    }
    a[pivot] = a[start];
    pivot = start;
  }
  a[pivot] = key;
  //小区间优化
  if (pivot - 1 - left < 10)
  {
    InsertSort(a+left, pivot - 1 - left + 1);
  }
  else
  {
    QuickSort(a, left, pivot - 1);
  }
  if (right - (pivot + 1) < 10)
  {
    InsertSort(a+pivot+1, right - (pivot + 1) + 1);
  }
  else
  {
    QuickSort(a, pivot + 1, right);
  }
}

1.2.2 前后指针法:

基本思想:

目的都是与挖坑法一样,就是将选出的数排到正确的位置,先保存最左边元素;然后定义left为prev,下一位为cur,从cur开始向后找小,找到后就将++prev的值与cur交换,直至找到right,最后再将prev与保存的关键值交换直到cur访问到最后一个元素。最后别忘了将保存的下标值与下标为prev的元素交换。

具体代码实现:

int PartSort2(int* a, int left, int right)
{
  int keyi = left;
  int prev = left;
  int cur = left + 1;;
  while (cur <= right)
  {
    while (a[cur] < a[keyi] && ++prev != cur)
    {
      Swap(&a[prev], &a[cur]);
    }
    cur++;
  }
  Swap(&a[keyi], &a[prev]);
  return prev;
}
void QuickSort2(int* a, int begin, int end)
{
  if (begin >= end)
  {
    return;
  }
  int keyi = PartSort2(a, begin, end);
  QuickSort2(a, begin, keyi - 1);
  QuickSort2(a, keyi + 1, end);
}

1.2.3 左右指针法

基本思想:

先保存最左边的值,从右边找比保存值小的元素,从左边找到比保存值大的元素,然后交换,直至相遇,再交换相遇点与保存值,将相遇点的下标返回。

具体代码实现:

int PartSort3(int* a, int left, int right)
{
  int keyi = left;
  while (left < right)
  {
    while (left < right && a[right] >= a[keyi])
    {
      right--;
    }
    while (left < right && a[left] <= a[keyi])
    {
      left++;
    }
    if (left < right)
    {
      Swap(&a[left], &a[right]);
    }
  }
  int meeti = left;
  Swap(&a[keyi], &a[right]);
  return meeti;
}
void QuickSort3(int* a, int begin, int end)
{
  if (begin >= end)
  {
    return;
  }
  int keyi = PartSort2(a, begin, end);
  QuickSort3(a, begin, keyi-1);
  QuickSort3(a, keyi + 1, end);
}


目录
相关文章
|
SQL Oracle 关系型数据库
|
4天前
|
人工智能 自然语言处理 文字识别
阿里云百炼Qwen3.7-Max简介:能力、优势、支持订阅计划参考
Qwen3.7-Max是阿里云百炼面向智能体时代推出的新一代旗舰模型,对标GPT-5.5、Claude Opus 4.7等闭源旗舰。该模型支持百万级token上下文窗口,具备顶级推理能力、多模态搜索与视觉理解增强、流式输出低延迟响应等核心优势,覆盖编程、办公、长周期自主执行等复杂场景。同时支持OpenAI接口兼容,便于系统快速迁移。用户可通过Token Plan团队或节省计划等订阅方式灵活调用,适合企业级高要求场景使用。
8517 37
阿里云百炼Qwen3.7-Max简介:能力、优势、支持订阅计划参考
|
4天前
|
缓存 测试技术 API
Qwen 3.7 Plus 与 Max 实测:性价比与多模态能力差异解析(2026)
2026 年 6 月 1 日,阿里悄无声息地发布了 Qwen 3.7 Plus,距 Qwen 3.7 Max 上线刚好 11 天。同样的 1M 上下文,同样的 35 小时自治上限。但价格才是头条:Plus 是 0.40/M输入,Max是 2.50/M——便宜约 6 倍——并且还能看图、看视频。Vision Arena 上 Plus 已经排到 #16。所以这周真正值得讨论的问题不是”要不要为视觉能力买单”,而是”Max 凭什么用 6 倍价格换来 2 个百分点的 benchmark 领先”。
|
4天前
|
JavaScript 定位技术 API
CodeGraph 爆火:编程 Agent 需要的不是更多上下文,而是一张提前画好的代码地图
CodeGraph 是一款爆火的本地代码智能工具,通过 tree-sitter 解析 AST 构建结构化知识图谱(存于 SQLite),为编程 Agent 提前生成“代码地图”。它显著降低 Agent 在中大型项目中的探索成本——实测工具调用减少71%、Token 降57%、速度提升46%,支持19+语言及主流框架路由识别,完全离线、无需 API Key。
617 3
CodeGraph 爆火:编程 Agent 需要的不是更多上下文,而是一张提前画好的代码地图
|
4天前
|
人工智能 运维 JavaScript
阿里云Qoder CN(原通义灵码)全解析 产品形态、版本划分与技术适配说明
在AI辅助开发与智能办公工具持续普及的当下,阿里云旗下原通义灵码正式更名为Qoder CN,同时延伸出QoderWork CN、Qoder CN CLI、Qoder CN Mobile等多款配套产品,形成覆盖代码开发、日常办公、终端交互、移动端使用的完整工具矩阵。Qoder CN核心定位为AI智能编码助手,深度适配主流代码编辑器、集成开发环境以及终端场景;QoderWork CN则偏向桌面端综合办公辅助,二者面向不同使用场景,划分了多个版本档位,搭配差异化资源配额、功能权限与计费规则,同时兼容多款主流大模型。
620 4
|
4天前
|
数据采集 人工智能 前端开发
让 Coding Agent 从黑盒到透明:阿里云 Agent 观测审计数据采集实践
AI Agent 规模化落地带来执行黑盒、行为难追溯、成本难度量三大难题。阿里云基于 OTel 标准,面向 Coding Agent、个人通用助理和框架型 Agent,推出 LoongSuite Pilot、插件及探针等无侵入采集方案,让 Agent 实现可看见、可分析、可审计、可治理。
709 148
|
4天前
|
人工智能 缓存 自然语言处理
阿里Qwen3.7-Max评测:Agent能力显著提升,耗时与调用成本大幅下降
阿里云百炼推出面向智能体的旗舰大模型Qwen3.7-Max,具备长周期自主执行能力,显著提升编程、办公自动化等复杂任务处理水平;支持MCP集成与多框架兼容,并以限时5折+100万Tokens免费试用大幅降低使用门槛,助力企业高效落地AI应用。在阿里云百炼平台快速体验:https://t.aliyun.com/U/fPVHqY
1949 10
|
4天前
|
存储 安全 Java
AgentScope Java 2.0:打造分布式、企业级智能体底座
AgentScope 2.0 面向分布式部署、稳定运行、权限安全等企业级需求全面升级,打造支持多租户隔离与长期稳定运行的企业级智能体底座。
|
4天前
|
人工智能 运维 API
2026年阿里云百炼通义千问Qwen3.7-plus深度介绍 功能特性、使用优势及618大促订阅方案指南
大模型技术的普及,让AI能力逐步融入个人办公、内容创作、代码编写、企业运营、教育培训等各类场景。不同定位的模型对应不同使用需求,旗舰级模型性能强劲但使用成本偏高,轻量化模型价格低廉却难以胜任复杂任务,而介于两者之间的中端主力模型,凭借均衡的能力、亲民的定价、广泛的场景适配性,成为绝大多数个人用户、小型团队、中小企业的首选。
742 1
|
4天前
|
人工智能 安全 定位技术
CodeGraph深度解析 让Claude Code工具调用直降七成的核心原理与实操教程
如今以Claude Code为代表的AI编程智能体已经成为开发者日常编码、项目重构、漏洞修复的必备工具。但在长期使用过程中,几乎所有开发者都会遇到同一个明显痛点:AI虽然具备强大的代码生成与分析能力,却常常陷入盲目探索的循环中。
1347 2