常用内部排序算法之四:简单选择排序、直接插入排序和冒泡排序

本文主要是介绍常用内部排序算法之四:简单选择排序、直接插入排序和冒泡排序,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言

之所以把这三类算法放在一块,是因为除此之外的算法都是在这三类算法的基础上进行优化的。简单选择排序的思想是每一趟 ni+1(i=1,2,...,n1) 个记录中选择最小的记录作为有序序列的第 i 个记录。直接插入排序的思想是将一个记录插入到已经排好序的有序序列中,从而得到一个新的、记录数增加1的有序表。冒泡排序的算法思想是不断在交换,通过交换完成最终的排序,每一趟的交换就会把最大的记录取出来,下一趟则会把第二大的记录取出来,这样每进行一趟交换就把一个记录取出来的过程称为冒泡。

简单选择排序算法

简单选择的排序的算法思想是:通过ni次关键字间的比较,从 ni+1 个记录中选出关键字最小的记录,并和第 i(1in) 个记录交换之。其算法代码如下:

package com.rhwayfun.algorithm.sort;public class SelectSort {public void selectSort(int[] a){int i,j,min;for (i = 0; i < a.length; i++) {//假设第一个位置的值是最小值min = i;for(j = i + 1; j < a.length; j++){if(a[min] > a[j]){min = j;}}//如果min不等于i,说明找到最小值的下标if(min != i){swap(a,i,min);}}for(i = 0; i < a.length; i++){System.out.println(a[i]);}}private void swap(int[] a, int i, int min) {int temp = a[i];a[i] = a[min];a[min] = temp;}public static void main(String[] args) {new SelectSort().selectSort(new int[]{9,1,5,8,3,7,4,6,2});}
}

观察代码可以发现,第 i 趟排序需要比较ni次关键字的比较,所以总共需要比较 n1i=1(ni)=n1+n2+...+1=n(n1)2 次。最好的情况下,交换0次,最差的情况是交换 n1 次,所以最终的时间复杂度是 O(n2)

直接插入排序算法

直接插入排序算法的思想是:将一个记录插入到已经排序的有序表中,从而得到一个新的、记录数增加1的有序表。其处理过程是,在排序刚开始的时候,把第一个元素当做是排序的记录,当依次插入后面的元素的时候,就获得其插入的位置,然后形成一个新的有序表。其算法代码如下:

package com.rhwayfun.algorithm.sort;public class InsertSort2 {public void insertSort(int[] a) {int i,j,temp;for(i = 1; i < a.length; i++){if(a[i] < a[i-1]){temp = a[i];for(j = i - 1; j >= 0 && a[j] > temp; j--){a[j+1] = a[j];}a[j+1] = temp;}}for(i = 0;i < a.length; i++){System.out.println(a[i]);}}public static void main(String[] args) {new InsertSort2().insertSort(new int[]{9,1,5,8,3,7,4,6,2});}
}

从空间上分析,直接插入排序算法只需要一个辅助空间。
从时间复杂度上分析,最好的情况是排序的记录本身是有序的,所以时间复杂度是 O(n) ;在最坏的情况,待排序的记录是逆序的,那么此时的时间复杂度是 O(n24) 。所以虽然量级仍然是 n2 ,但是直接插入排序算法的时间复杂度是优于冒泡排序算法和简单选择排序的。

冒泡排序

冒泡排序的基本思想是两两比较相邻记录的关键字,如果反序就交换,直到没有反序的关键字为止。下面是一种实现思路:

package com.rhwayfun.algorithm.sort;public class BubbleSort3 {public void bubbleSort(int[] a){int i,j;for(i = 0; i < a.length; i++){for(j = i + 1; j < a.length; j++){if(a[i] > a[j]){swap(a,i,j);}}}for(i = 0; i < a.length; i++){System.out.println(a[i]);}}private void swap(int[] a, int i, int j) {int temp = a[i];a[i] = a[j];a[j] = temp;}public static void main(String[] args) {new BubbleSort3().bubbleSort(new int[]{9,1,5,8,3,7,4,6,2});}
}

这种版本也是我第一时间写出来的,但是可以发现一个问题,在排好第一个和第二个为止之后,数字3反而被排到了最后面。下面是针对这种情况的改良版代码:

public void bubbleSort2(int[] a){int i,j;for(i = 0; i < a.length; i++){for(j = a.length - 2; j >= i; j--){if(a[j] > a[j + 1]){swap(a,j,j+1);}}}for(i = 0; i < a.length; i++){System.out.println(a[i]);}}

这里的改进主要把第二个for循环由从前往后比较改成由后往前进行比较了,这样的好处是可以把本来较小的元素放在尽可能前一点的位置,这种差异性在数据量较大的时候能够体现出来。以上改良版的冒泡排序使用于基本无序的序列,如果是基本有序的序列再使用上述的算法进行排序就会出现一个问题:那就是可能在进行完前几次的冒泡之后就已经是有序的了,那么后面的冒泡都是多余的。下面得代码是针对这种情况进行的优化:

public void bubbleSort3(int[] a){int i,j;boolean flag = true;for(i = 0; i < a.length && flag; i++){flag = false;for(j = a.length - 2; j >= i; j--){if(a[j] > a[j + 1]){//如果不进行数据交换,说明是有序的swap(a,j,j+1);flag = true;}}}for(i = 0; i < a.length; i++){System.out.println(a[i]);}}

如果在面试中要求写出冒泡排序算法的代码,写最后一种情况就可以了。

下面分析冒泡排序算法的时间复杂度:在最坏的情况就是待排序的记录是逆序的,此时的时间复杂度是 O(n2) ;最好的情况是,排序表本身就是有序的,那么在这种情况下,时间复杂度是 O(n)

这篇关于常用内部排序算法之四:简单选择排序、直接插入排序和冒泡排序的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1016577

相关文章

springboot项目中常用的工具类和api详解

《springboot项目中常用的工具类和api详解》在SpringBoot项目中,开发者通常会依赖一些工具类和API来简化开发、提高效率,以下是一些常用的工具类及其典型应用场景,涵盖Spring原生... 目录1. Spring Framework 自带工具类(1) StringUtils(2) Coll

openCV中KNN算法的实现

《openCV中KNN算法的实现》KNN算法是一种简单且常用的分类算法,本文主要介绍了openCV中KNN算法的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的... 目录KNN算法流程使用OpenCV实现KNNOpenCV 是一个开源的跨平台计算机视觉库,它提供了各

Java String字符串的常用使用方法

《JavaString字符串的常用使用方法》String是JDK提供的一个类,是引用类型,并不是基本的数据类型,String用于字符串操作,在之前学习c语言的时候,对于一些字符串,会初始化字符数组表... 目录一、什么是String二、如何定义一个String1. 用双引号定义2. 通过构造函数定义三、St

springboot+dubbo实现时间轮算法

《springboot+dubbo实现时间轮算法》时间轮是一种高效利用线程资源进行批量化调度的算法,本文主要介绍了springboot+dubbo实现时间轮算法,文中通过示例代码介绍的非常详细,对大家... 目录前言一、参数说明二、具体实现1、HashedwheelTimer2、createWheel3、n

Mybatis 传参与排序模糊查询功能实现

《Mybatis传参与排序模糊查询功能实现》:本文主要介绍Mybatis传参与排序模糊查询功能实现,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧... 目录一、#{ }和${ }传参的区别二、排序三、like查询四、数据库连接池五、mysql 开发企业规范一、#{ }和${ }传参的

使用Python开发一个简单的本地图片服务器

《使用Python开发一个简单的本地图片服务器》本文介绍了如何结合wxPython构建的图形用户界面GUI和Python内建的Web服务器功能,在本地网络中搭建一个私人的,即开即用的网页相册,文中的示... 目录项目目标核心技术栈代码深度解析完整代码工作流程主要功能与优势潜在改进与思考运行结果总结你是否曾经

Mysql表的简单操作(基本技能)

《Mysql表的简单操作(基本技能)》在数据库中,表的操作主要包括表的创建、查看、修改、删除等,了解如何操作这些表是数据库管理和开发的基本技能,本文给大家介绍Mysql表的简单操作,感兴趣的朋友一起看... 目录3.1 创建表 3.2 查看表结构3.3 修改表3.4 实践案例:修改表在数据库中,表的操作主要

springboot简单集成Security配置的教程

《springboot简单集成Security配置的教程》:本文主要介绍springboot简单集成Security配置的教程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,... 目录集成Security安全框架引入依赖编写配置类WebSecurityConfig(自定义资源权限规则

SpringBoot实现MD5加盐算法的示例代码

《SpringBoot实现MD5加盐算法的示例代码》加盐算法是一种用于增强密码安全性的技术,本文主要介绍了SpringBoot实现MD5加盐算法的示例代码,文中通过示例代码介绍的非常详细,对大家的学习... 目录一、什么是加盐算法二、如何实现加盐算法2.1 加盐算法代码实现2.2 注册页面中进行密码加盐2.

Java时间轮调度算法的代码实现

《Java时间轮调度算法的代码实现》时间轮是一种高效的定时调度算法,主要用于管理延时任务或周期性任务,它通过一个环形数组(时间轮)和指针来实现,将大量定时任务分摊到固定的时间槽中,极大地降低了时间复杂... 目录1、简述2、时间轮的原理3. 时间轮的实现步骤3.1 定义时间槽3.2 定义时间轮3.3 使用时