探究排序算法:比较与非比较排序算法及性能分析

探究排序算法:比较与非比较排序算法及性能分析

排序算法是计算机科学中的基本问题,它涉及将一组元素按照特定的顺序排列。本文将深入介绍比较排序算法和非比较排序算法,包括每个算法的原理、Java代码示例以及它们的性能分析和比较。

比较排序算法

1. 冒泡排序(Bubble Sort)

原理:冒泡排序通过多次遍历数组,比较相邻元素并交换,使较大的元素逐渐“冒泡”到数组的尾部。

代码示例

public class BubbleSort {public static void bubbleSort(int[] array) {int n = array.length;for (int i = 0; i < n - 1; i++) {for (int j = 0; j < n - i - 1; j++) {if (array[j] > array[j + 1]) {int temp = array[j];array[j] = array[j + 1];array[j + 1] = temp;}}}}public static void main(String[] args) {int[] array = {64, 34, 25, 12, 22, 11, 90};bubbleSort(array);System.out.println("冒泡排序结果:");for (int num : array) {System.out.print(num + " ");}}
}

2. 插入排序(Insertion Sort)

原理:插入排序将数组分为已排序和未排序两部分,逐个将未排序元素插入到已排序部分的正确位置。

代码示例

public class InsertionSort {public static void insertionSort(int[] array) {int n = array.length;for (int i = 1; i < n; i++) {int key = array[i];int j = i - 1;while (j >= 0 && array[j] > key) {array[j + 1] = array[j];j--;}array[j + 1] = key;}}public static void main(String[] args) {int[] array = {12, 11, 13, 5, 6};insertionSort(array);System.out.println("插入排序结果:");for (int num : array) {System.out.print(num + " ");}}
}

3. 选择排序(Selection Sort)

原理:选择排序在未排序部分找到最小元素,然后与已排序部分的末尾元素交换,逐步构建有序序列。

代码示例

public class SelectionSort {public static void selectionSort(int[] array) {int n = array.length;for (int i = 0; i < n - 1; i++) {int minIndex = i;for (int j = i + 1; j < n; j++) {if (array[j] < array[minIndex]) {minIndex = j;}}int temp = array[minIndex];array[minIndex] = array[i];array[i] = temp;}}public static void main(String[] args) {int[] array = {64, 25, 12, 22, 11};selectionSort(array);System.out.println("选择排序结果:");for (int num : array) {System.out.print(num + " ");}}
}

4. 希尔排序(Shell Sort)

原理:希尔排序是插入排序的改进,通过将数组分为多个子序列来进行排序,逐渐减小步长,最终使整个数组基本有序。

代码示例

public class ShellSort {public static void shellSort(int[] array) {int n = array.length;for (int gap = n / 2; gap > 0; gap /= 2) {for (int i = gap; i < n; i++) {int temp = array[i];int j = i;while (j >= gap && array[j - gap] > temp) {array[j] = array[j - gap];j -= gap;}array[j] = temp;}}}public static void main(String[] args) {int[] array = {12, 34, 54, 2, 3};shellSort(array);System.out.println("希尔排序结果:");for (int num : array) {System.out.print(num + " ");}}
}

5. 快速排序(Quick Sort)

原理:快速排序使用分治法,选择一个基准元素,将数组分成左右两部分,左边的元素小于基准,右边的元素大于基准,然后对左右子数组递归地应用快速排序。

代码示例

public class QuickSort {public static void quickSort(int[] array, int low, int high) {if (low < high) {int pivotIndex = partition(array, low, high);quickSort(array, low, pivotIndex - 1);quickSort(array, pivotIndex + 1, high);}}public static int partition(int[] array, int low, int high) {int pivot = array[high];int i = low - 1;for (int j = low; j < high; j++) {if (array[j] < pivot) {i++;int temp = array[i];array[i] = array[j];array[j] = temp;}}int temp = array[i + 1];array[i + 1] = array[high];array[high] = temp;return i + 1;}public static void main(String[] args) {int[] array = {10, 7, 8, 9, 1, 5};int n = array.length;quickSort(array, 0, n - 1);System.out.println("快速排序结果:");for (int num : array) {System.out.print(num + " ");}}
}

6. 归并排序(Merge Sort)

原理:归并排序是一种分治算法,将数组逐步分成两个子数组,分别排序后再合并成一个有序数组。

代码示例

public class MergeSort {public static void mergeSort(int[] array, int left, int right) {if (left < right) {int mid = (left + right) / 2;mergeSort(array, left, mid);mergeSort(array, mid + 1, right);merge(array, left, mid, right);}}public static void merge(int[] array, int left, int mid, int right) {int n1 = mid - left + 1;int n2 = right - mid;int[] leftArray = new int[n1];int[] rightArray = new int[n2];for (int i = 0; i < n1; i++) {leftArray[i] = array[left + i];}for (int j = 0; j < n2; j++) {rightArray[j] = array[mid + 1 + j];}int i = 0, j = 0, k = left;while (i < n1 && j < n2) {if (leftArray[i] <= rightArray[j]) {array[k] = leftArray[i];i++;} else {array[k] = rightArray[j];j++;}k++;}while (i < n1) {array[k] = leftArray[i];i++;k++;}while (j < n2) {array[k] = rightArray[j];j++;k++;}}public static void main(String[] args) {int[] array = {12, 11, 13, 5, 6, 7};int n = array.length;mergeSort(array, 0, n - 1);System.out.println("归并排序结果:");for (int num : array) {System.out.print(num + " ");}}
}

非比较排序算法

1. 计数排序(Counting Sort)

原理:计数排序适用于元素范围较小的情况,它统计每个元素出现的次数,然后根据统计信息重新构建有序序列。

代码示例

public class CountingSort {public static void countingSort(int[] array) {int n = array.length;int max = Arrays.stream(array).max().getAsInt();int min = Arrays.stream(array).min().getAsInt();int range = max - min + 1;int[] count = new int[range];int[] output = new int[n];for (int num : array) {count[num - min]++;}for (int i = 1; i < range; i++) {count[i] += count[i - 1];}for (int i = n - 1; i >= 0; i--) {output[count[array[i] - min] - 1] = array[i];count[array[i] - min]--;}for (int i = 0; i < n; i++) {array[i] = output[i];}}public static void main(String[] args) {int[] array = {4, 2, 2, 8, 3, 3, 1};countingSort(array);System.out.println("计数排序结果:");for (int num : array) {System.out.print(num + " ");}}
}

2. 桶排序(Bucket Sort)

原理:桶排序将数据分成多个桶,每个桶内部使用其他排序算法(通常是插入排序),然后将桶内有序的数据合并起来。

代码示例

public class BucketSort {public static void bucketSort(double[] array) {int n = array.length;List<Double>[] buckets = new List[n];for (int i = 0; i < n; i++) {buckets[i] = new ArrayList<>();}for (int i = 0; i < n; i++) {int bucketIndex = (int) (n * array[i]);buckets[bucketIndex].add(array[i]);}for (int i = 0; i < n; i++) {Collections.sort(buckets[i]);}int index = 0;for (int i = 0; i < n; i++) {for (double num : buckets[i]) {array[index++] = num;}}}public static void main(String[] args) {double[] array = {0.42, 0.32, 0.33, 0.52, 0.37, 0.47, 0.51};bucketSort(array);System.out.println("桶排序结果:");for (double num : array) {System.out.print(num + " ");}}
}

3. 基数排序(Radix Sort)

原理:基数排序逐位对元素进行排序,先按最低位排序,再按次低位排序,直到最高位。

代码示例

public class RadixSort {public static void radixSort(int[] array) {int n = array.length;int max = Arrays.stream(array).max().getAsInt();for (int exp = 1; max / exp > 0;exp *= 10) {countingSortByDigit(array, n, exp);}}public static void countingSortByDigit(int[] array, int n, int exp) {int[] output = new int[n];int[] count = new int[10];for (int i = 0; i < n; i++) {count[(array[i] / exp) % 10]++;}for (int i = 1; i < 10; i++) {count[i] += count[i - 1];}for (int i = n - 1; i >= 0; i--) {output[count[(array[i] / exp) % 10] - 1] = array[i];count[(array[i] / exp) % 10]--;}for (int i = 0; i < n; i++) {array[i] = output[i];}}public static void main(String[] args) {int[] array = {170, 45, 75, 90, 802, 24, 2, 66};radixSort(array);System.out.println("基数排序结果:");for (int num : array) {System.out.print(num + " ");}}
}

排序算法的性能分析和比较

排序算法的性能在不同场景下会有所不同。比较排序算法的平均时间复杂度如下:

  • 冒泡排序:O(n^2)
  • 插入排序:O(n^2)
  • 选择排序:O(n^2)
  • 希尔排序:取决于步长序列,一般在 O(n log n) 到 O(n^2) 之间
  • 快速排序:O(n log n) 平均,最坏情况为 O(n^2)
  • 归并排序:O(n log n)
  • 堆排序:O(n log n)

非比较排序算法的时间复杂度如下:

  • 计数排序:O(n + k),其中 k 是数据范围
  • 桶排序:O(n^2) 最坏情况,平均情况较好
  • 基数排序:O(n * k),其中 k 是数字的位数

综合来看,在大多数情况下,快速排序和归并排序是较优的选择,具有较快的平均性能。计数排序和基数排序在特定情况下也能表现优异。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/53006.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

WPF读取dicom序列:实现上一帧、下一帧、自动播放、暂停

一、整体设计概况 创建WPF程序使用.Net Framework4.8定义Image控件展示图像增加标签展示dcm文件信息规划按钮触发对应的事件:上一帧、下一帧、自动播放、暂停、缩放、播放速率二、页面展示 三、代码逻辑分析 Windows窗体加载Loaded事件:生成初始图像信息Windows窗体加载Mous…

电脑入门:宽带路由器常见故障排除技巧

宽带路由器在企业网络中的应用是相当广泛的,在运行的过程中出现故障是在所难免的,虽然故障现象多种多样,引起故障发生的原因也不尽相同,但从大体上可以把这些故障分为硬件故障和软件故障,具体来说就是一些网络连接性问题、配置文件选项问题以及网络协议问题等。 由于路由器…

【Docker】Docker网络与存储(三)

前言&#xff1a; Docker网络与存储的作用是实现容器之间的通信和数据持久化&#xff0c;以便有效地部署、扩展和管理容器化应用程序。 文章目录 Docker网络桥接网络容器之间的通信 覆盖网络创建一个覆盖网络 Docker存储卷 总结 Docker网络 Docker网络是在容器之间提供通信的机…

el-tree树回显删除某项,再次点开树形组件无变化,实际数据已改变

el-tree树回显删除某项&#xff0c;再次点开树形组件无变化&#xff0c;实际数据已改变 页面有添加和删除已选选项的按钮&#xff0c;点击删除一个选项&#xff0c;再点添加&#xff0c;打开树形弹窗&#xff0c;发现弹窗被删除的选项还在 原因&#xff1a; 发现是添加的时候&…

leetcode做题笔记​101. 对称二叉树

给你一个二叉树的根节点 root &#xff0c; 检查它是否轴对称。 思路一&#xff1a;递归 bool isSymmetric(struct TreeNode* root){if (root NULL) return true;return fun(root->left, root->right); }int fun(struct TreeNode* l_root, struct TreeNode* r_root) {…

Android中的APK打包与安全

aapt2命令行实现apk打包 apk文件结构 classes.dex&#xff1a;Dex&#xff0c;即Android Dalvik执行文件 AndroidManifest.xml&#xff1a;工程中AndroidManifest.xml编译后得到的二进制xml文件 META-INF&#xff1a;主要保存各个资源文件的SHA1 hash值&#xff0c;用于校验…

paddlenlp安装踩坑记录

错误1 ModuleNotFoundError: No module named paddle.metric我下载paddlepaddle-gpu2.5.0.post117解决了&#xff0c;最开始下载的2.5.1报错&#xff0c;post后面的117是我的cuda版本&#xff0c;要写你对应的版本号 python3 -m pip install paddlepaddle-gpu2.5.0.post117 -…

HTTPS单向认证与双向认证

HTTPS单向认证与双向认证 HTTPSCA证书单向认证双向认证 HTTPS Https就是HTTPSSL/TSL的简称。 SSL(Secure Socket Layer 安全套接层)是TCP/IP协议中基于HTTP之下TCP之上的一个可选协议层。 起初HTTP在传输数据时使用的是明文&#xff0c;传输过程中并不安全。网景&#xff08;N…

SMC_TRAFO_GantryCutter2 (FB) 带刀片旋向龙门

裁布机&#xff1a;刀片按XY走向&#xff0c;偏转刀片角度。 pi&#xff1a;目标位置矢量&#xff08;x&#xff0c;y&#xff09;&#xff0c;插值器的输出 v&#xff1a;当前路径切线的矢量&#xff0c;插值器的输出 dOffsetX&#xff1a; x轴的附加偏移 dOffsetY&#xf…

桥梁安全监测方法和内容是什么?

桥梁安全监测方法和内容是什么?桥梁监测是保障桥梁安全和稳定的重要手段。随着科技的进步&#xff0c;桥梁监测技术和设备不断完善&#xff0c;监测内容也越来越全面。本文万宾科技小编将为大家介绍桥梁安全监测的方法和内容&#xff0c;以期帮助大家更好地了解这一领域。 桥梁…

kafka--技术文档--spring-boot集成基础简单使用

阿丹&#xff1a; 查阅了很多资料了解到&#xff0c;使用了spring-boot中整合的kafka的使用是被封装好的。也就是说这些使用其实和在linux中的使用kafka代码的使用其实没有太大关系。但是逻辑是一样的。这点要注意&#xff01; 使用spring-boot整合kafka 1、导入依赖 核心配…

华为质量管理:从产品质量到用户体验,Kano模型成为新方向

目录 前言 华为质量管理的四个阶段 基于 IPD 如何做质量管理呢&#xff1f; CSDN相关课程 作者简介 前言 今天继续来谈谈华为流程体系中的质量管理过程。 通常来说质量具体是指产品的质量&#xff0c;也就是产品的使用价值及其属性。 产品再细分的话可以分为三个层次&a…

知识图谱Neo4j安装到实践全过程

前言&#xff1a; Hello大家好&#xff0c;我是Dream。 在本次实战中&#xff0c;我们将一起完成知识图谱Neo4j安装到实践全过程&#xff0c;探索其中的关系和属性。知识图谱是一种以三元组形式存储的数据结构&#xff0c;由实体、关系和属性组成&#xff0c;能够帮助我们更好地…

STC8H8K64U单片机PWM配置

STC8H8K64U单片机配置PWM的通道1、通道2、通道7、通道8。STC8可以分为两组PWM&#xff0c;PWMA和PWMB&#xff0c;在使用时&#xff0c;如果同时使用了两组&#xff0c;那么两组的寄存器需要同时配置。 一、PWM输出频率计算公式 边沿对齐 PWM输出频率 系统工作频率 / (PWMx…

Python数据分析 | 各种图表对比总结

本期将带领大家一起对在数据可视化的过程中常用的一些图表进行下总结&#xff1a; 条形图 【适用场景】 适用场合是二维数据集&#xff08;每个数据点包括两个值x和y&#xff09;&#xff0c;但只有一个维度需要比较&#xff0c;用于显示一段时间内的数据变化或显示各项之间的…

【洛谷】P2440 木材加工

原题链接&#xff1a;https://www.luogu.com.cn/problem/P2440 1. 题目描述 2. 思路分析 整体思路&#xff1a;二分答案 设置一个变量longest来记录最长木头的长度&#xff0c;sum记录切成的小段数量之和。 令左边界l0&#xff0c;右边界llongest。 写一个bool类型的check…

华为OD机试 - 采样过滤(Java JS Python)

题目描述 在做物理实验时,为了计算物体移动的速率,通过相机等工具周期性的采样物体移动距离。 由于工具故障,采样数据存在误差甚至错误的情况。 需要通过一个算法过滤掉不正确的采样值。 不同工具的故障模式存在差异,算法的各类门限会根据工具类型做相应的调整。 请实…

mysql 、sql server trigger 触发器

sql server mySQL create trigger 触发器名称 { before | after } [ insert | update | delete ] on 表名 for each row 触发器执行的语句块## 表名&#xff1a; 表示触发器监控的对象 ## before | after : 表示触发的时间&#xff0c;before : 表示在事件之前触发&am…

4.14 tcp_tw_reuse 为什么默认是关闭的?

开启 tcp_tw_reuse 参数可以快速复用处于 TIME_WAIT 状态的 TCP 连接时&#xff0c;相当于缩短了 TIME_WAIT 状态的持续时间。 tcp_tw_reuse 是什么&#xff1f; TIME_WAIT 状态的持续时间是 60 秒&#xff0c;这意味着这 60 秒内&#xff0c;客户端一直会占用着这个端口。端…

iOS HealthKit 介绍

文章目录 一、简介二、权限配置1. 在开发者账号中勾选HealthKit2. 在targets的capabilities中添加HealthKit。3. infoPlist需要配置权限 三、创建健康数据管理类1. 引入头文件2. 健康数据读写权限3. 检查权限4. 读取步数数据5. 写入健康数据 四、运行获取权限页面 一、简介 He…