【算法基础】选择排序算法 - JAVA
一、算法基础
1.1 什么是选择排序
选择排序是一种简单直观的排序算法,它的工作原理是:首先在未排序序列中找到最小(或最大)元素,存放到排序序列的起始位置,然后再从剩余未排序元素中继续寻找最小(或最大)元素,然后放到已排序序列的末尾。以此类推,直到所有元素均排序完毕。
1.2 选择排序的基本思想
选择排序的核心思想是:
- 将输入数据分为已排序区域和未排序区域
- 不断从未排序区域选择最小元素,放入已排序区域的末尾
- 重复上述过程直到全部排序完成
1.3 时间复杂度
选择排序的时间复杂度分析如下:
- 最好情况时间复杂度:O(n²)
- 最坏情况时间复杂度:O(n²)
- 平均时间复杂度:O(n²)
无论输入数据是否已经部分排序,选择排序始终需要进行 n(n-1)/2 次比较,因此其时间复杂度总是 O(n²)。
1.4 空间复杂度
选择排序是一种原地排序算法,它只需要一个用于交换的临时变量,因此空间复杂度为 O(1)。
二、Java实现
2.1 基础实现
public class SelectionSort {public static void selectionSort(int[] arr) {int n = arr.length;// 外层循环控制已排序区域的边界for (int i = 0; i < n - 1; i++) {// 找出未排序区域中的最小值索引int minIndex = i;for (int j = i + 1; j < n; j++) {if (arr[j] < arr[minIndex]) {minIndex = j;}}// 将找到的最小值与未排序区域的第一个元素交换if (minIndex != i) {int temp = arr[i];arr[i] = arr[minIndex];arr[minIndex] = temp;}}}// 测试代码public static void main(String[] args) {int[] arr = {64, 25, 12, 22, 11};selectionSort(arr);System.out.println("排序后的数组:");for (int i : arr) {System.out.print(i + " ");}}
}
2.2 算法步骤解析
- 初始状态:整个数组视为未排序区域
- 第一次迭代:
- 在整个数组中找到最小元素
- 将其与数组第一个元素交换位置
- 数组第一个元素现在位于正确位置
- 第二次迭代:
- 从第二个元素开始,在剩余未排序元素中找到最小值
- 将其与数组第二个元素交换位置
- 重复过程:继续执行,直到所有元素都排序完毕
三、选择排序的特点
3.1 优点
- 实现简单,易于理解和编码
- 交换次数少:每次内循环只需要进行一次交换操作,最多进行 n-1 次交换
- 原地排序:不需要额外的存储空间
- 稳定性可控:可以实现为稳定排序算法(通过特定实现方式)
3.2 缺点
- 时间效率低:无论输入如何,时间复杂度始终为 O(n²)
- 不适合大规模数据:当数据量大时,性能显著下降
- 不会提前终止:即使数组已经排序,算法仍会执行完所有步骤
四、算法优化方案
4.1 双向选择排序
双向选择排序在每次迭代中同时查找最小值和最大值,减少了循环次数:
public static void bidirectionalSelectionSort(int[] arr) {int left = 0;int right = arr.length - 1;while (left < right) {int minIndex = left;int maxIndex = left;// 找出最小值和最大值for (int i = left + 1; i <= right; i++) {if (arr[i] < arr[minIndex]) {minIndex = i;}if (arr[i] > arr[maxIndex]) {maxIndex = i;}}// 将最小值放到左边if (minIndex != left) {int temp = arr[left];arr[left] = arr[minIndex];arr[minIndex] = temp;// 如果最大值是left,那么交换后最大值位置变为minIndexif (maxIndex == left) {maxIndex = minIndex;}}// 将最大值放到右边if (maxIndex != right) {int temp = arr[right];arr[right] = arr[maxIndex];arr[maxIndex] = temp;}left++;right--;}
}
4.2 使用堆数据结构
可以使用最小堆来优化选择排序:
- 构建一个最小堆 O(n)
- 依次取出堆顶元素 O(log n)
这种方法实际上就是堆排序,时间复杂度降低到 O(n log n)。
五、实例分析
5.1 示例数组排序过程
以数组 [64, 25, 12, 22, 11]
为例:
- 第一次迭代:
- 找到最小值 11(索引 4)
- 交换11和64:
[11, 25, 12, 22, 64]
- 已排序:[11],未排序:[25, 12, 22, 64]
- 第二次迭代:
- 在剩余部分找到最小值 12(索引 2)
- 交换12和25:
[11, 12, 25, 22, 64]
- 已排序:[11, 12],未排序:[25, 22, 64]
- 第三次迭代:
- 在剩余部分找到最小值 22(索引 3)
- 交换22和25:
[11, 12, 22, 25, 64]
- 已排序:[11, 12, 22],未排序:[25, 64]
- 第四次迭代:
- 在剩余部分找到最小值 25(索引 3)
- 不需要交换(已在正确位置)
- 已排序:[11, 12, 22, 25],未排序:[64]
- 排序完成:
[11, 12, 22, 25, 64]
5.2 性能分析
对比不同输入规模的性能表现:
- 小规模数据(n ≤ 50):选择排序性能可接受
- 中等规模数据(50 < n ≤ 1000):性能明显下降
- 大规模数据(n > 1000):性能严重下降,不推荐使用
六、应用场景
6.1 适用场景
- 小规模数据排序:当数据量较小时,选择排序简单易实现
- 对交换操作敏感的场景:选择排序的交换次数最多为 n-1 次
- 辅助教学:作为排序算法的基础教学示例
- 内存受限的嵌入式系统:实现简单且空间复杂度低
6.2 不适用场景
- 大规模数据排序:性能太低,应选择更高效的算法
- 几乎已排序的数据:选择排序无法利用数据的部分有序性
七、总结
选择排序是一种简单直观的排序算法,核心思想是不断从未排序区域中选择最小元素放入已排序区域。
核心要点:
- 时间复杂度始终为 O(n²),无论输入如何
- 空间复杂度为 O(1),是一种原地排序算法
- 交换次数较少,最多进行 n-1 次交换
- 实现简单,代码易于理解和编写
- 对于大型数据集不推荐使用
选择排序虽然不是最高效的排序算法,但它的简单性和低空间复杂度使其在特定场景下仍有价值。对于编程初学者来说,理解选择排序有助于掌握排序算法的基本概念及其实现方法。
相关文章:
【算法基础】选择排序算法 - JAVA
一、算法基础 1.1 什么是选择排序 选择排序是一种简单直观的排序算法,它的工作原理是:首先在未排序序列中找到最小(或最大)元素,存放到排序序列的起始位置,然后再从剩余未排序元素中继续寻找最小…...
电商平台的流量秘密:代理IP在用户行为分析中的角色
在电商江湖中,流量是氧气,用户行为数据是DNA。当你在电商平台点击商品、加入购物车时,背后有一套精密的系统正在分析你的每个动作。而在这套系统的运作中,代理IP正扮演着"隐形推手"的角色——它既是数据采集的"隐身…...
批量清洗与修改 YOLO 标签:删除与替换指定类别
在使用 YOLO 格式的数据进行训练或部署前,常常需要对标签文件进行清洗或修改。本文整理了两种常见场景的 Python 脚本:删除指定类别 和 修改某类为其他类,并支持自动打印检测到该类别的文件名,帮助你快速定位问题数据。 …...

Python项目源码57:数据格式转换工具1.0(csv+json+excel+sqlite3)
1.智能路径处理:自动识别并修正文件扩展名,根据转换类型自动建议目标路径,实时路径格式验证,自动补全缺失的文件扩展名。 2.增强型预览功能:使用pandastable库实现表格预览,第三方模块自己安装一下&#x…...
TypeScript 中,属性修饰符
在 TypeScript 中,属性修饰符(Property Modifiers)是用于修饰类的属性或方法的关键字,它们可以改变属性或方法的行为和访问权限。TypeScript 提供了三种主要的属性修饰符:public、private 和 protected。此外ÿ…...

雷赛伺服电机
ACM0经济 编码器17位: ACM1基本 编码器23位磁编, ACM2通用 编码器24位光电, 插头定义:...
基础编程题目集 6-8 简单阶乘计算
本题要求实现一个计算非负整数阶乘的简单函数。 函数接口定义: int Factorial( const int N ); 其中N是用户传入的参数,其值不超过12。如果N是非负整数,则该函数必须返回N的阶乘,否则返回0。 裁判测试程序样例: #in…...

【deepseek教学应用】001:deepseek如何撰写教案并自动实现word排版
本文讲述利用deepseek如何撰写教案并自动实现word高效完美排版。 文章目录 一、访问deepseek官网二、输入教案关键词三、格式转换四、word进一步排版 一、访问deepseek官网 官网:https://www.deepseek.com/ 进入主页后,点击【开始对话】,如…...

CH32V208GBU6沁恒绑定配对获取静态地址
从事嵌入式单片机的工作算是符合我个人兴趣爱好的,当面对一个新的芯片我即想把芯片尽快搞懂完成项目赚钱,也想着能够把自己遇到的坑和注意事项记录下来,即方便自己后面查阅也可以分享给大家,这是一种冲动,但是这个或许并不是原厂希望的,尽管这样有可能会牺牲一些时间也有哪天原…...
【C/C++】RPC与线程间通信:高效设计的关键选择
文章目录 RPC与线程间通信:高效设计的关键选择1 RPC 的核心用途2 线程间通信的常规方法3 RPC 用于线程间通信的潜在意义4 主要缺点与限制4.1 缺点列表4.2 展开 5 替代方案6 结论 RPC与线程间通信:高效设计的关键选择 在C或分布式系统设计中,…...
跨线程和跨进程通信还有多种方式对比
📊 常见通信机制对比 通信方式跨线程支持跨进程支持同步/异步性能编程复杂度特点与适用场景SendMessage✅✅(同桌面)同步较高(阻塞)低简单窗口通信、控制PostMessage✅✅(同桌面)异步高低通知、事件触发COM/DCOM✅✅同步/异步中中高系统级服务、进程间服务封装Socket✅…...

RT Thread Studio创建软件和硬件RTC工程
MCU型号:STM32F103RET6 一.配置软件模拟RTC 1.生成一个带串口输出的工程文件,新建RT-Thread项目工程文件。 2.查看电路图中的串口输出管脚,根据STMCubeMx软件可知此串口为USART1,选择芯片型号为STM32F103RET6,控制台…...
Oracle EBS AP发票被预付款核算创建会计科目时间超长
背景 由于客户职能部门的水电、通信和物业等等费用统一管理或对接部门报销费,在报销费的时候,用户把所有费用分摊到各个末级部门,形成AP发票行有上千行, 问题症状 1、用户过账时,请求创建会计科目一直执行20多个小时未完成,只能手工强行取消请求。 2、取消请求以后,从后…...
驱动开发硬核特训 · Day 30(下篇): 深入解析 lm48100q I2C 音频编解码器驱动模型(基于 i.MX8MP)
作者:嵌入式Jerry 视频教程请关注 B 站:“嵌入式Jerry” 一、背景与目标 在本篇中,我们围绕 TI 的 lm48100q 音频编解码器 展开,深入讲解其作为 I2C 外设如何集成至 Linux 内核音频子系统(ASoC)࿰…...
运维--计划任务
计划任务分为一次性和循环性的计划任务 1.date的用法 date 月日时分年 eg:date 100118222023 date:查看时间和日期、修改时间和日期 获取当前日期:date +%F F:日期 获取当前时间:date +%H:%M:%S H:时 M:分 S:秒 获取周几: date +%w w:周 …...

苍穹外卖心得体会
1 登录认证 技术点:JWT令牌技术(JSON Web Token) JWT(JSON Web Token)是一种令牌技术,主要由三部分组成:Header头部、Payload载荷和Signature签名。Header头部存储令牌的类型(如JW…...
Python爬虫实战:获取jd商城最新5060ti 16g显卡销量排行榜商品数据并做分析,为显卡选购做参考
一、引言 1.1 研究目的 本研究旨在利用 Python 爬虫技术,从京东商城获取 “5060ti 16g” 型号显卡的商品数据,并对这些数据进行深入分析。具体目标包括: 实现京东商城的模拟登录,突破登录验证机制,获取登录后的访问权限。高效稳定地爬取按销量排名前 20 的 “5060ti 16g…...
Fedora升级Google Chrome出现GPG check FAILED问题解决办法
https://dl.google.com/linux/linux_signing_key.pub 的 GPG 公钥(0x7FAC5991)已安装 https://dl.google.com/linux/linux_signing_key.pub 的 GPG 公钥(0xD38B4796)已安装 仓库 "google-chrome" 的 GPG 公钥已安装,但是不适用于此软件包。 请检查此仓库的…...
信息系统监理师第二版教材模拟题第三组(含解析)
信息系统监理师模拟题第三组(30题) 监理基础理论 信息系统工程监理的性质是( ) A. 服务性、独立性、公正性、科学性 B. 强制性、营利性、行政性、技术性 C. 临时性、从属性、随意性、主观性 D. 单一性、封闭性、被动性、保守性答案:A 解析:监理具有服务性、独立性、公正…...

Zcanpro搭配USBCANFD-200U在新能源汽车研发测试中的应用指南(周立功/致远电子)
——国产工具链的崛起与智能汽车测试新范式 引言:新能源汽车测试的国产化突围 随着新能源汽车智能化、网联化程度的提升,研发测试面临三大核心挑战:多协议融合(CAN FD/LIN/以太网)、高实时性数据交互需求、复杂工况下…...

青少年抑郁症患者亚群结构和功能连接耦合的重构
目录 1 研究背景及目的 2 研究方法 2.1 数据来源与参与者 2.1.1 MDD患者: 2.1.2 健康对照组: 2.2 神经影像分析流程 2.2.1 图像采集与预处理: 2.2.2 网络构建: 2.2.3 区域结构-功能耦合(SC-FC耦合)…...

SQL手工注入(DVWA)
手工SQL注入攻击的标准思路 Low等级 (1)判断是否存在注入 (2)猜解字段个数 (3)确定字段顺序 (4)获取当前数据库 (5)获取当前数据库中的表 (…...

【大模型系列篇】Qwen3开源全新一代大语言模型来了,深入思考,更快行动
Qwen3开源模型全览 Qwen3是全球最强开源模型(MoEDense) Qwen3 采用混合专家(MoE)架构,总参数量 235B,激活仅需 22B。 Qwen3 预训练数据量达 36T,并在后训练阶段多轮强化学习,将非思…...
第 11 届蓝桥杯 C++ 青少组中 / 高级组省赛 2020 年真题,选择题详细解释
一、选择题 第 2 题 在二维数组按行优先存储的情况下,元素 a[i][j] 前的元素个数计算如下: 1. **前面的完整行**:共有 i 行,每行 n 个元素,总计 i * n 个元素。 2. **当前行的前面元素**:在行内&#x…...

flutter 专题 一百零四 Flutter环境搭建
Flutter简介 Flutter 是Google开发的一个移动跨平台(Android 和 iOS)的开发框架,使用的是 Dart 语言。和 React Native 不同的是,Flutter 框架并不是一个严格意义上的原生应用开发框架。Flutter 的目标是用来创建高性能、高稳定性…...
Android之Button、ImageButton、ChipGroup用法
一 控件名称及UI代码 Button、ImageButton、ChipGroup <?xml version="1.0" encoding="utf-8"?> <androidx.coordinatorlayout.widget.CoordinatorLayoutxmlns:android="http://schemas.android.com/apk/res/android"xmlns:app=&qu…...
【AI提示词】二八法则专家
提示说明 精通二八法则(帕累托法则)的广泛应用,擅长将其应用于商业、管理、个人发展等领域,深入理解其在不同场景中的具体表现和实际意义。 提示词 # Role: 二八法则专家## Profile - language: 中文 - description: 精通二八法…...

玩玩OCR
一、Tesseract: 1.下载windows版: tesseract 2. 安装并记下路径,等会要填 3.保存.py文件 import pytesseract from PIL import Image def ocr_local_image(image_path):try:pytesseract.pytesseract.tesseract_cmd rD:\Programs\Tesseract-OCR\tesse…...
set autotrace报错
报错: SQL> set autotrace traceonly SP2-0618: Cannot find the Session Identifier. Check PLUSTRACE role is enabled SP2-0611: Error enabling STATISTICS report原因分析: 根据上面的错误提示“SP2-0618: Cannot find the Session Identifie…...
C++如何设计和实现缓存(cache)来减少对后端存储的访问压力
随着数据量的激增和用户对低延迟、高吞吐量需求的不断提升,如何减少系统瓶颈、提升响应速度成为了开发者的核心挑战之一。在这一背景下,缓存(cache)作为一种关键的技术手段,逐渐成为解决性能问题的核心策略。缓存的本质是通过存储频繁访问的数据或计算结果,减少对后端存储…...