常见的初级排序算法,这次全搞懂

开发 前端 算法
相信所有的程序员刚开始接触到的算法都会是排序算法,因为排序在对数据处理和计算有这重要的地位,排序算法往往是其他算法的基础;本文我们就先从初级排序算法开始学习算法。

[[383742]]

本文转载自微信公众号「贝塔学JAVA」,作者Silently9527。转载本文请联系贝塔学JAVA公众号。

本文已被Github仓库收录 https://github.com/silently9527/JavaCore

程序员常用的IDEA插件:https://github.com/silently9527/ToolsetIdeaPlugin

完全开源的淘客项目:https://github.com/silently9527/mall-coupons-server

前言

相信所有的程序员刚开始接触到的算法都会是排序算法,因为排序在对数据处理和计算有这重要的地位,排序算法往往是其他算法的基础;本文我们就先从初级排序算法开始学习算法。

排序算法的模板

在开始之前我们先定义一个排序算法通用的模板,在后面的排序算法都会实现这个模板

  1. public interface SortTemplate { 
  2.  
  3.     void sort(Comparable[] array); 
  4.  
  5.     default void print(Comparable[] array) { 
  6.         for (Comparable a : array) { 
  7.             System.out.print(a + " "); 
  8.         } 
  9.     } 
  10.  
  11.     default boolean less(Comparable a, Comparable b) { 
  12.         return a.compareTo(b) < 0; 
  13.     } 
  14.  
  15.     default void exch(Comparable[] array, int i, int j) { 
  16.         Comparable tmp = array[i]; 
  17.         array[i] = array[j]; 
  18.         array[j] = tmp; 
  19.     } 
  20.      
  • Comparable: 为了让我们实现的排序算法更加的通用,可以排序任意的对象,所以我们这里使用了Comparable数组
  • sort: 不同的排序算法实现的方式不一样,子类自己去实现
  • less: 定义的公用方法,如果a < b就返回true
  • exch: 定义的公用方法,交换数组中的两个对象
  • print: 打印出数据中的每个元素

选择排序

算法实现的思路:

  • 首先找到数组中的最小元素,
  • 其实将它和数组中的第一个元素进行交换,这样就排定了一个元素;
  • 再次找出剩余元素中最小的元素与数组中的第二个元素进行交换,如此反复直到所有元素都是有序的

代码实现:

  1. public class SelectionSort implements SortTemplate { 
  2.  
  3.     @Override 
  4.     public void sort(Comparable[] array) { 
  5.         int length = array.length; 
  6.         for (int i = 0; i < length; i++) { 
  7.             int min = i; 
  8.             for (int j = i + 1; j < length; j++) { 
  9.                 if (less(array[j], array[min])) { 
  10.                     min = j; 
  11.                 } 
  12.             } 
  13.             exch(array, i, min); 
  14.         } 
  15.     } 
  16.  

假如输入的数组是有序的,我们会发现选择排序运行的时候和未排序的时间一样长!

对于N个元素的数组,使用「选择排序的时间复杂度是O(n2)」

选择排序的是「数据移动最少」的,交换的次数与数组的大小是线性关系,N个元素的数组需要N次交换

冒泡排序

算法实现的思路:

比较相邻的两个元素,如果前一个比后一个大,那么就交换两个元素的位置

对每一组相邻的元素执行同样的操作,直到最后一个元素,操作完成之后就可以排定一个最大的元素

如此往复,直到数组中所有的元素都有序

代码实现:

  1. public class BubbleSort implements SortTemplate { 
  2.  
  3.     @Override 
  4.     public void sort(Comparable[] array) { 
  5.         int length = array.length - 1; 
  6.         for (int i = 0; i < length; i++) { 
  7.             for (int j = 0; j < length - i; j++) { 
  8.                 if (less(array[j + 1], array[j])) { 
  9.                     exch(array, j, j + 1); 
  10.                 } 
  11.             } 
  12.         } 
  13.     } 
  14.  

对于N个元素的数组,使用「冒泡排序的时间复杂度是O(n2)」

插入排序

想象我们在玩扑克牌时,整理扑克牌都是把每一张插入到左边已经排好序的牌中适当的位置。插入排序的思路类似

算法实现的思路:

  • 初始默认第一个元素就是有序的,当前索引的位置从0开始
  • 先后移动当前索引的位置,当前索引位置左边的元素是有序的,从后往前开始扫码与当前索引位置元素进行比较
  • 当确定当前索引位置上的元素在左边有序适合的位置之后,插入到该位置上
  • 如果当确定当前索引位置上的元素大于了已排序的最后一个元素,那么当前索引位置直接往后移动
  • 如此反复,直到所有元素有序

代码实现:

  1. public class InsertionSort implements SortTemplate { 
  2.  
  3.     @Override 
  4.     public void sort(Comparable[] array) { 
  5.         int length = array.length; 
  6.         for (int i = 1; i < length; i++) { 
  7.             for (int j = i; j > 0 && less(array[j], array[j - 1]); j--) { 
  8.                 exch(array, j, j - 1); 
  9.             } 
  10.         } 
  11.     } 
  12.  

从代码的实现我们可以看出,当遇到了当前索引的元素大于了左边有序数组的最后一个元素时,内层循环就直接结束了,所以所我们排序的数组中存在着部分有序,那么插入排序算法会很快。

考虑最糟糕的情况,如果输入数组是一个倒置的,那么插入排序的效率和选择排序一样,「时间复杂度是O(n2)」

希尔排序

对于大规模的乱序数组插入排序很慢,是因为它只交换相邻的元素,元素只能一点一点的从数组中移动到正确的位置;插入排序对于部分有序的数组排序是的效率很高;

希尔排序基于这两个特点对插入排序进行了改进;

算法实现的思路

  • 首先设置一个步长h用来分隔出子数组
  • 用插入排序将h个子数组独立排序
  • 减小h步长继续排序子数组,直到h步长为1
  • 当步长为1时就成了普通的插入排序,这样数组一定是有序的

希尔排序高效的原因,在排序之初,各个子数组都很短,子数组排序之后都是部分有序的,这两种情况都很适合插入排序。

代码实现:

  1. public class ShellSort implements SortTemplate { 
  2.  
  3.     @Override 
  4.     public void sort(Comparable[] array) { 
  5.         int gap = 1; 
  6.         int length = array.length; 
  7.  
  8.         while (gap < length / 3) { 
  9.             gap = 3 * gap + 1; 
  10.         } 
  11.  
  12.         while (gap >= 1) { 
  13.             for (int i = gap; i < length; i++) { 
  14.                 for (int j = i; j >= gap && less(array[j], array[j - gap]); j -= gap) { 
  15.                     exch(array, j, j - gap); 
  16.                 } 
  17.             } 
  18.             gap = gap / 3; 
  19.         } 
  20.     } 
  21.  

 

责任编辑:武晓燕 来源: 贝塔学JAVA
相关推荐

2021-02-22 07:29:07

算法初级排序

2021-02-03 15:30:10

面试垃圾回收器前端

2022-05-17 12:23:25

排序算法面试

2021-01-29 17:07:26

排序算法数组

2017-11-22 14:20:07

前端JavaScript排序算法

2023-12-04 07:49:06

选择排序排序算法

2022-03-26 08:49:13

MySQL数据存储

2011-02-17 09:11:40

JavaScript算法

2022-11-01 18:29:25

Go语言排序算法

2020-10-23 10:10:59

Promise前端代码

2021-09-30 07:57:13

排序算法面试

2022-09-24 09:03:55

前端单元测试冒泡排序

2021-09-04 23:40:53

算法程序员前端

2016-09-30 14:23:16

数据结构算法八大排序算法

2023-02-09 07:39:01

2021-11-10 09:17:18

程序员排序算法搜索算法

2021-11-08 15:12:48

排序算法面试

2024-08-30 14:34:00

2021-04-07 20:01:23

Go变量常量

2016-10-14 23:16:50

数据排序数据原始数据
点赞
收藏

51CTO技术栈公众号