选择排序的实现

简介: 选择排序的实现

目录

前言

什么是选择排序

选择排序的实现

基本思想

具体实现代码

选择排序的原理

选择排序的性能

时间复杂度

稳定性

改进


前言

上期我们对插入排序中的直接插入排序和希尔排序进行了分析,我们了解了如何实现它们和它们的性能比较,知道了希尔排序是直接插入排序的优化。今天我们来学习选择排序。

什么是选择排序

百度百科中说:排序定义。所谓计算机中的排序,就是使一串记录,按照其中的某个或某些关键字的大小,递增或递减的排列起来的操作。而排序算法则是一种能将一串数据依照特定的方式进行排列的一种算法。排序方式。利用所需重排记录的排序码的值的大小,按照升序或降序将原纪录的顺序重新安排。排序类别。内排序可以分为插入排序、选择排序、交换排序、归并排序以及分配排序。选择排序法是一种不稳定的排序算法。它的工作原理是每一次从待排序的数据元素中选出最小(或最大)的一个元素,存放在序列的起始位置,然后,再从剩余未排序元素中继续寻找最小(大)元素,然后放到已排序序列的末尾。以此类推,直到全部待排序的数据元素排完。常见的选择排序可以分为直接选择排序、树状排序以及堆排序。今天我们说的就是直接选择排序

选择排序的实现

基本思想

首先在未排序的数组中找到最小元素,存放到排序数组的起始位置。

再从剩余未排序元素中继续寻找最小元素,然后放到已排序数组的末尾。

重复第二步,直到所有元素全部排序完毕。

具体实现代码

#include <stdio.h>
//直接选择排序
void SelectSort(int* arr, int len)
{
  int i = 0;
  int j = 0;
  //这里是它比较的趟数,就是每个元素都和后面的比较,找出最小值
  //到最后一个元素的时候已经是有序的了所以就是len-1次
  for (i = 0; i < len - 1; i++)
  {
    //把未处理的数组中第一个元素的下标假设为最小下标
    int min = i;
    //找出无序数组中最小的元素
    for (j = i + 1; j < len; j++)
    {
      //min下标的元素大于后面的元素
      if (arr[min] > arr[j])
      {
        //min跟换下标
        min = j;
      }
    }
    //将无序数组中的第一个元素与真正最小值交换
    int tmp = arr[min];
    arr[min] = arr[i];
    arr[i] = tmp;
  }
}
int main()
{
  //待排序数组
  int arr[] = { 2,3,1,0,4, 9,10,6,5 };
  //直接选择排序
  SelectSort(arr, sizeof(arr) / sizeof(arr[0]));
  //打印
  int i = 0;
  for (i = 0; i < sizeof(arr) / sizeof(arr[0]); i++)
  {
    printf("%d ", arr[i]);
  }
  return 0;
}

选择排序的原理

选择排序法的第一层循环从起始元素开始选到倒数第二个元素,主要是在每次进入的第二层循环之前,将外层循环的下标赋值给临时变量,接下来的第二层循环中,如果发现有比这个最小位置处的元素更小的元素,则将那个更小的元素的下标赋给临时变量,最后,在二层循环退出后,如果临时变量改变,则说明,有比当前外层循环位置更小的元素,需要将这两个元素交换 。这里我们以代码中的数组为例:

第一次:我们可以将数组中第一个元素暂时设为最小值,然后与后面的元素相比较,3大于最小值2跳过,1小于2则最小值给到2,再和后面比较发现后面的元素都大于1.这时就将1与数组第一个元素交换。第二次:我们将第二个元素为最小值与后面的元素比较,发现3大于2,则最小值就是2,2再与后面的元素比较发现没有比他小的,则将2放在已经排好数组的后面。第三次:假设最小值为3,发现后面的都比他大,不变。第4次:假设4为最小值,发现后面的元素都大于它,不变。第5次:假设9为最小值,发现后面的元素10大于它,不变。6小于它,则最小值为6,再比较,发现5小于6,则6为最小值,将6放到有序数组的后面,后面的步骤重复如此,直到排序成功。

选择排序的性能

时间复杂度

选择排序的交换操作介于 0 和 (n - 1) 次之间。选择排序的比较操作为 n (n - 1) / 2 次之间。选择排序的赋值操作介于 0 和 3 (n - 1) 次之间。比较次数O(n^2),比较次数与关键字的初始状态无关,交换次数O(n),最好情况是,已经有序,交换0次;最坏情况交换n-1次,逆序交换n/2次。交换次数比冒泡排序少多了,由于交换所需CPU时间比比较所需的CPU时间多,n值较小时,选择排序比冒泡排序快。其中直接选择排序的时间复杂度为O(n*n),空间复杂度为O(1)。树形选择排序的时间复杂度为O(nlog2n),空间复杂度为O(n)。堆排序的平均时间复杂度为O(nlog2n),效率高,但是实现相对复杂,空间代价为O(1)。

稳定性

选择排序是给每个位置选择当前元素最小的,比如给第一个位置选择最小的,在剩余元素里面给第二个元素选择第二小的,依次类推,直到第n-1个元素,第n个元素不用选择了,因为只剩下它一个最大的元素了。那么,在一趟选择,如果一个元素比当前元素小,而该小的元素又出现在一个和当前元素相等的元素后面,那么交换后稳定性就被破坏了。举个例子,序列5 8 5 2 9,我们知道第一遍选择第1个元素5会和2交换,那么原序列中两个5的相对前后顺序就被破坏了,所以选择排序是一个不稳定的排序算法 。

改进

二元选择排序

改进思路: 简单选择排序,每趟循环只能确定一个元素排序后的定位。根据之前冒泡排序的经验,我们可以考虑改进为每趟循环确定两个元素(当前趟最大和最小记录)的位置,从而减少排序所需的循环次数。改进后对n个数据进行排序,最多只需进行[n/2]趟循环即可。

堆排序

堆排序是一种树形选择排序,是对直接选择排序的有效改进。具体的分析我们留到后面讲堆排序时再详细说明。

这些改进再后面的文章会再次提到的。

目录
相关文章
|
JavaScript 前端开发
Vue3--Vue Router详解--学习笔记
Vue3--Vue Router详解--学习笔记
311 3
如何写出一份优秀的技术简历?
如何写出一份优秀的技术简历?
217 0
|
4月前
|
人工智能 数据库 云计算
🔥躺着赚佣金!阿里云推荐计算巢服务让你轻松"带货"云计算
阿里云推出“推荐服务赚佣金”计划,无需技术背景,分享链接即可轻松赚取额外收入。高达20%佣金,200+精选服务任选,实时追踪收益,适合个人推广的数字副业!
413 9
|
9月前
|
人工智能 自然语言处理 API
销售易NeoCRM与Salesforce:优势特色大比拼
本文对比了销售易NeoCRM与Salesforce两款CRM系统。销售易NeoCRM功能涵盖销售、客户、营销管理等,具AI赋能和移动办公优势,界面现代化,价格灵活适合中小企业;Salesforce功能全面,AI平台强大,生态系统丰富,全球化支持出色,适合大型及跨国企业。两者各有优劣,企业应根据自身需求选择合适的CRM系统,以实现高效管理和业务增长。
|
SQL 测试技术 数据库
【Python】已解决:pymssql引发的MSSQLDatabaseException错误
【Python】已解决:pymssql引发的MSSQLDatabaseException错误
563 0
|
关系型数据库 MySQL OLAP
快速入门:搭建你的第一个AnalyticDB实例
【10月更文挑战第25天】在大数据时代,高效的在线分析处理(OLAP)成为企业决策的关键。AnalyticDB是阿里云推出的一款完全托管的实时数据仓库服务,它能够支持PB级的数据量和高并发的查询需求。作为一名数据工程师,我有幸在工作中使用了AnalyticDB,并积累了丰富的实践经验。本文将从个人角度出发,详细介绍如何快速搭建你的第一个AnalyticDB实例,包括创建实例、连接数据库、导入数据和执行简单查询等步骤。
504 0
|
Web App开发
Chrome——谷歌浏览器chrome如何模拟其他客户端
Chrome——谷歌浏览器chrome如何模拟其他客户端
343 1
Chrome——谷歌浏览器chrome如何模拟其他客户端
|
Kubernetes Cloud Native Linux
云原生入门:Kubernetes的简易部署与应用
【8月更文挑战第49天】在云原生的世界里,Kubernetes(K8s)是一颗璀璨的星。本文将带你走进K8s的世界,从安装到简单应用,轻松驾驭这个强大的容器编排工具。让我们一起探索云原生的奥秘,解锁新技能!
|
存储 NoSQL 关系型数据库
深入探索地理空间查询:如何优雅地在MySQL、PostgreSQL及Redis中实现精准的地理数据存储与检索技巧
深入探索地理空间查询:如何优雅地在MySQL、PostgreSQL及Redis中实现精准的地理数据存储与检索技巧
3067 0
|
前端开发 Shell 容器
前端练习小项目——视觉冲击卡片
前端练习小项目——视觉冲击卡片