高快省的排序算法

有没有既不浪费空间又可以快一点的排序算法呢?那就是“快速排序”啦!光听这个名字是不是就觉得很高端呢。

假设我们现在对“6 1 2 7 9 3 4 5 10 8”这个10个数进行排序。首先在这个序列中随便找一个数作为基准数(不要被这个名词吓到了,就是一个用来参照的数,待会你就知道它用来做啥的了)。为了方便,就让第一个数6作为基准数吧。接下来,需要将这个序列中所有比基准数大的数放在6的右边,比基准数小的数放在6的左边,类似下面这种排列:

3 1 2 5 4 6 9 7 10 8

在初始状态下,数字6在序列的第1位。我们的目标是将6挪到序列中间的某个位置,假设这个位置是k。现在就需要寻找这个k,并且以第k位为分界点,左边的数都小于等于6,右边的数都大于等于6。想一想,你有办法可以做到这点吗?

排序算法显神威

方法其实很简单:分别从初始序列“6 1 2 7 9 3 4 5 10 8”两端开始“探测”。先从右往左找一个小于6的数,再从左往右找一个大于6的数,然后交换他们。这里可以用两个变量i和j,分别指向序列最左边和最右边。我们为这两个变量起个好听的名字“哨兵i”和“哨兵j”。刚开始的时候让哨兵i指向序列的最左边(即i=1),指向数字6。让哨兵j指向序列的最右边(即=10),指向数字。

首先哨兵j开始出动。因为此处设置的基准数是最左边的数,所以需要让哨兵j先出动,这一点非常重要(请自己想一想为什么)。哨兵j一步一步地向左挪动(即j–),直到找到一个小于6的数停下来。接下来哨兵i再一步一步向右挪动(即i++),直到找到一个数大于6的数停下来。最后哨兵j停在了数字5面前,哨兵i停在了数字7面前。

现在交换哨兵i和哨兵j所指向的元素的值。交换之后的序列如下:

6 1 2 5 9 3 4 7 10 8

到此,第一次交换结束。接下来开始哨兵j继续向左挪动(再友情提醒,每次必须是哨兵j先出发)。他发现了4(比基准数6要小,满足要求)之后停了下来。哨兵i也继续向右挪动的,他发现了9(比基准数6要大,满足要求)之后停了下来。此时再次进行交换,交换之后的序列如下:

6 1 2 5 4 3 9 7 10 8

第二次交换结束,“探测”继续。哨兵j继续向左挪动,他发现了3(比基准数6要小,满足要求)之后又停了下来。哨兵i继续向右移动,糟啦!此时哨兵i和哨兵j相遇了,哨兵i和哨兵j都走到3面前。说明此时“探测”结束。我们将基准数6和3进行交换。交换之后的序列如下:

3 1 2 5 4 6 9 7 10 8

到此第一轮“探测”真正结束。此时以基准数6为分界点,6左边的数都小于等于6,6右边的数都大于等于6。回顾一下刚才的过程,其实哨兵j的使命就是要找小于基准数的数,而哨兵i的使命就是要找大于基准数的数,直到i和j碰头为止。

OK,解释完毕。现在基准数6已经归位,它正好处在序列的第6位。此时我们已经将原来的序列,以6为分界点拆分成了两个序列,左边的序列是“3 1 2 5 4”,右边的序列是“9 7 10 8”。接下来还需要分别处理这两个序列。因为6左边和右边的序列目前都还是很混乱的。不过不要紧,我们已经掌握了方法,接下来只要模拟刚才的方法分别处理6左边和右边的序列即可。现在先来处理6左边的序列现吧。

左边的序列是“3 1 2 5 4”。请将这个序列以3为基准数进行调整,使得3左边的数都小于等于3,3右边的数都大于等于3。好了开始动笔吧

如果你模拟的没有错,调整完毕之后的序列的顺序应该是:

2 1 3 5 4

OK,现在3已经归位。接下来需要处理3左边的序列“2 1”和右边的序列“5 4”。对序列“2 1”以2为基准数进行调整,处理完毕之后的序列为“1 2”,到此2已经归位。序列“1”只有一个数,也不需要进行任何处理。至此我们对序列“2 1”已全部处理完毕,得到序列是“1 2”。序列“5 4”的处理也仿照此方法,最后得到的序列如下:

1 2 3 4 5 6 9 7 10 8

对于序列“9 7 10 8”也模拟刚才的过程,直到不可拆分出新的子序列为止。最终将会得到这样的序列,如下

1 2 3 4 5 6 7 8 9 10

到此,排序完全结束。细心的同学可能已经发现,快速排序的每一轮处理其实就是将这一轮的基准数归位,直到所有的数都归位为止,排序就结束了。下面上个霸气的图来描述下整个算法的处理过程。

这是为什么呢?

快速排序之所比较快,因为相比冒泡排序,每次交换是跳跃式的。每次排序的时候设置一个基准点,将小于等于基准点的数全部放到基准点的左边,将大于等于基准点的数全部放到基准点的右边。这样在每次交换的时候就不会像冒泡排序一样每次只能在相邻的数之间进行交换,交换的距离就大的多了。因此总的比较和交换次数就少了,速度自然就提高了。当然在最坏的情况下,仍可能是相邻的两个数进行了交换。因此快速排序的最差时间复杂度和冒泡排序是一样的都是O(N2),它的平均时间复杂度为O(NlogN)。其实快速排序是基于一种叫做“二分”的思想。我们后面还会遇到“二分”思想,到时候再聊。先上代码,如下

public classQuickSort {public static void quickSort(int[] arr,int low,inthigh) {inti,j,temp,t;if(low >high) {return;

}

i=low;

j=high;//temp 是基准位

temp =arr[low];while (i

while (temp <= arr[j] && i

j--;

}while (temp >= arr[i] && i

i++;

}//交换

if(i

t=arr[j];

arr[j]=arr[i];

arr[i]=t;

}

}//此时 i 和 j 是相等的//最后将基准为与i和j相等位置的数字交换

arr[low] =arr[i];

arr[i]=temp;//递归调用左半数组

quickSort(arr,low,j-1);//递归调用右半数组

quickSort(arr, j+1, high);

}public static voidmain(String[] args){int[] arr = {6,1,2,7,9,3,4,5,10,8};

quickSort(arr,0,arr.length -1);for (int i = 0; i < arr.length; i++) {

System.out.println(arr[i]);

}

}

}

java快排原理_快速排序原理及实现(java)相关推荐

  1. java sort算法名称_快速排序算法(Quick Sort)(java)

    /** * 快速排序算法是基于分治策略的一种排序算法,下面是一个递归的快速排序. * @author liuy */ public class QuickSort { public static vo ...

  2. java快排原理_Java数据结构与算法——快速排序

    声明:码字不易,转载请注明出处,欢迎文章下方讨论交流. 前言:Java数据结构与算法专题会不定时更新,欢迎各位读者监督.本篇文章介绍排序算法中最常用也是面试中最容易考到的排序算法--快排,包括快排的思 ...

  3. 数据结构之排序【归并排序和快排的顶级优化和快排的三种原理的实现及分析】 内含动态演示图

    文章目录 引言: 1.归并排序(MergeSort) 2.快速排序的优化(顶级优化) 3.快速排序的三种思路的代码实现及分析 4.归并排序和快排第3原理的测试 引言: 刚刚去回顾了一下递归实现的几个小 ...

  4. 关键词(快排)刷词原理和方法

    关键词(快排)刷词原理和方法 刷词原理Seo理论:主关键词的相关关键词流量占比越大,排名越好.(长尾词带动主词排名现象的解释) 举例:如果说我们想做的主关键词是seo,那么可以统计出每天在百度上搜索和 ...

  5. 算法——java快排(快速排序)

    手写java快排(快速排序)代码--双边循环法,本人亲测哈哈!! package com.abc.backend.algo;import java.util.Arrays;/*** 快排(双边循环法) ...

  6. java快排算法解读,java 快排的思路与算法

    java 快排的思路与算法 有时候面试的时候的会问道Arrays.sort()是怎么实现的,我以前根本不知道是什么东西,最近点进去看了一下.直接吓傻, //看到这个时候还是比较淡定的,可怕的事情来了. ...

  7. java 快排_八大排序-快速排序(搞定面试之手写快排)

    概要 快速排序由C. A. R. Hoare在1960年提出,是八大排序算法中最常用的经典排序算法之一.其广泛应用的主要原因是高效,核心算法思想是分而治之.快速排序经常会被作为面试题进行考察,通常的考 ...

  8. java 快排_秋招|字节跳动Java后台已上岸,发个面经回馈牛油

    作者:beatLAL 链接:https://www.nowcoder.com/discuss/215891 来源:牛客网 本人从牛客大佬的面经中学到了很多,这次上岸回馈牛油,把自己面经分享给大家,希望 ...

  9. java 快排_百度在年前会在打击一轮快排!

    百度近期借助线上搜索知识公开课对外通知,百度搜索即将在年前再来一波快排站大清扫,不论是不是烟雾弹站长们最近还是要低调,以免被误伤. 根据百度搜索线上公开课内容大纲了解到,这次课程包括了,新站外链,网站 ...

最新文章

  1. Python使用matplotlib可视化气泡图、并使用encircle函数自定义多边形圈定可视化图像中的指定区域(Bubble plot with Encircling)
  2. java 导出文件,导出多个文件方案~
  3. c# mysql fill_C#里sqlDataAdapter.fill(DataSet,String)的用法
  4. jenkins java_具有WildFly,Arquillian,Jenkins和OpenShift的Java EE 7部署管道
  5. Analyzer报表结果行
  6. 开源Blog系统-欧式风格家具网站源码v1.5.4
  7. idea查看ruby代码_Java代码审计入门篇:WebGoat 8(初见)
  8. 注意:ORACLE 11G ADG RAC 这个情况下并不能高可用
  9. python读取文档中有很多指标的数据 写成矩阵_图像处理与特征提取 —— 从 MATLAB 到 Python(一)图像、矩阵与数据的读写...
  10. 关于Python局部变量和全局变量必须知道的几句话
  11. centos directory server
  12. 奔图打印机显示未连接_打印机无法打印的10种解决方法
  13. Java程序-自制班级随机选人程序
  14. Linux权限中x是什么意思,linux里的drwxr-xr-x代表的意思
  15. WebP图片格式处理和兼容使用
  16. 单片机程序跑飞的原因
  17. python设置刻度间隔不等_Matplotlib绘图遇到时间刻度就犯难?现在,一次性告诉你四种方法...
  18. c语言高精度加减法程序,C语言实现高精度加减法
  19. 根据excel模板导出excel
  20. 发一份魔兽改键工具的代码,vc6.0环境,带聊天模式识别功能

热门文章

  1. java 堆外内存_详解Java堆外内存
  2. MongoDB基本查询语句
  3. 读计算机操作系统的读后感,《计算机操作系统》读后感锦集
  4. Android OpglEs录屏功能实现
  5. c语言强制转换字符类型,C语言数据类型转换实例代码
  6. 爬取某电影网站排行榜TOP250
  7. Ribbon重试的坑
  8. 一行代码实现任意网页视频配速播放
  9. 汇编语言笔记(全)(长文警告)
  10. Vue项目打包部署教程及常见错误-前端开发