让你一看就懂的快速排序算法(Java)
快速排序
你也许会被快速排序的文章弄得迷迷糊糊,其实大体上去看,快速排序就一步:找个数做基准数,把小于它的数移到它左边,把大于它的数移到它右边。这句话是核心。然后我们只需要让基准数左边的重复上面的步骤,让基准数右边的再重复上面的步骤就完了。
比如我们有一个数组:
    int[] nums = {5, 2, 6, 8, 4, 7, 9, 1};
快速排序的思想就是使用递归,其实使用递归并不是多么复杂,在理解算法的思想后,只需要关注算法中重复的步骤,那就是递归的核心代码。
比如快速排序的算法思想,大体代码如下:
public void quick(int left, int right) {
    /*
    * 给这段代码起个名字为:基准校验
    * 把小于基准数的移到左边,把大于基准数的移到右边
    */
    quick(left, now); //继续处理左边的
    quick(now, right); //继续处理右边的
}
经过一遍基准校验,我们就找到了该基准数在完全排序后的正确位置! 
大体算法的流程图:

写出上面的大体算法步骤,就表示我们已经有了雏形,现在该如何实现找个数做基准数,把小于它的数移到它左边,把大于它的数移到它右边呢?
排除不满足的情况:
        //已经不满足条件就代表可以不用递归了
        if (left > right) {
            return;
        }
我们可以定义两个指针,用于遍历数组:
        int start = left;//起点下标
        int end = right;//终点下标
再找一个基准数:
        int temp = nums[left];//把第一个数作为基准点
遍历代码:
 /*如果左右指针还没有走到一起,代表还有位置没有遍历*/
 while (start != end) {
     //右指针先走,找到小于基准数的停止
     while (start < end && nums[end] >= temp) {
         end--;     //这是往左移动指针
     }
     //左指针后走,找到大于基准数的停止
     while (start < end && nums[start] <= temp) {
         start++;  //这是往右移动指针
     }
     //如果左右指针在未相遇时都找到了目标,则交换位置
     if (start < end) {
         int i = nums[start];
         nums[start] = nums[end];
         nums[end] = i;
     }
     //左右指针走到一起,则遍历结束
 }
 //把基准数与该点交换位置,因为这就是基准数的正确位置
 nums[left] = nums[start];
 nums[start] = temp;
如果还是不太清楚,不如自己动手试试!
拷贝完整代码:
    //定义一个数组
    static int[] nums = {5, 2, 6, 8, 4, 7, 9, 1};
    static int n = nums.length - 1;
    /**
     * 递归的数据结构就是栈
     * left right代表该段数组的起点和终点
     */
    public void quick(int left, int right) {
        //已经不满足条件就可以不用递归了
        if (left > right) {
            return;
        }
        //定义俩指针 用于移动
        int start = left;//起点下标
        int end = right;//终点下标
        int temp = nums[left];//把第一个数作为基准点
        pri(left, right);//打印此时的结果,不用在意
        while (start != end) {    //如果左右指针还没有走到一起,代表还有位置没有遍历
            while (start < end && nums[end] >= temp) { //右指针先走,找到小于基准数的停止
                end--;            //这是往左在移动指针
            }
            while (start < end && nums[start] <= temp) { //左指针后走,找到大于基准数的停止
                start++;     //这是往右在移动指针
            }
            if (start < end) {    //如果左右指针在未相遇时都找到了目标,则交换位置
                int i = nums[start];
                nums[start] = nums[end];
                nums[end] = i;
            }
        }
        //此时的left和right走到了一起
        //把基准数与该点交换位置
        nums[left] = nums[start];
        nums[start] = temp;
        prin(start);//打印输出,不用在意
        //以上代码的作用就是把小于基准数的移到左边,把大于基准数的移到右边
        quick(left, start - 1); //继续处理左边的,这里是一个递归的过程
        quick(start + 1, right); //继续处理右边的 ,这里是一个递归的过程
    }
    /**
    * 主程序入口
    */
    public static void main(String[] args) {
        new Test().quick(0, n);
    }
    /**
     * 以下代码忽略即可,用于打印输出
     */
    private void pri(int start, int end) {
        StringBuffer s = new StringBuffer();
        s.append("对数组 [");
        while (start <= end) {
            s.append(nums[start] + " ");
            start++;
        }
        s.append("]");
        s.append(" 排序");
        System.out.print(s);
    }
    private void prin(int j) {
        StringBuffer s = new StringBuffer();
        s.append(", 排序后 [");
        int start = 0;
        while (start <= n) {
            if (start == j) {
                s.append("(" + nums[start] + ") ");
            } else {
                s.append(nums[start] + " ");
            }
            start++;
        }
        s.append("]");
        s.append(" ");
        System.out.println(s);
    }
打印输出:
对数组 [5 2 6 8 4 7 9 1 ] 排序, 排序后 [4 2 1 (5) 8 7 9 6 ]
对数组 [4 2 1 ] 排序, 排序后 [1 2 (4) 5 8 7 9 6 ]
对数组 [1 2 ] 排序, 排序后 [(1) 2 4 5 8 7 9 6 ]
对数组 [2 ] 排序, 排序后 [1 (2) 4 5 8 7 9 6 ]
对数组 [8 7 9 6 ] 排序, 排序后 [1 2 4 5 6 7 (8) 9 ]
对数组 [6 7 ] 排序, 排序后 [1 2 4 5 (6) 7 8 9 ]
对数组 [7 ] 排序, 排序后 [1 2 4 5 6 (7) 8 9 ]
对数组 [9 ] 排序, 排序后 [1 2 4 5 6 7 8 (9) ] 												
											让你一看就懂的快速排序算法(Java)的更多相关文章
- 快速排序算法 java 实现
		
快速排序算法 java 实现 快速排序算法Java实现 白话经典算法系列之六 快速排序 快速搞定 各种排序算法的分析及java实现 算法概念 快速排序是C.R.A.Hoare于1962年提出的一种划分 ...
 - 排序算法系列:快速排序算法JAVA版(靠谱、清晰、真实、可用、不罗嗦版)
		
在网上搜索算法的博客,发现一个比较悲剧的现象非常普遍: 原理讲不清,混乱 啰嗦 图和文对不上 不可用,甚至代码还出错 为了不误人子弟耽误时间,推荐看一些靠谱的资源,如[啊哈!算法]系列: https: ...
 - 排序系列 之 快速排序算法 —— Java实现
		
基本思想: 通过一趟排序将要排序的数据分割成独立的两部分,其中一部分的所有数据都比另外一部分的所有数据都要小,然后再按此方法对这两部分数据分别进行快速排序,整个排序过程可以递归进行,以此达到整个数据变 ...
 - 【排序算法】快速排序算法 Java实现
		
快速排序是C.R.A.Hoare于1962年提出的一种划分交换排序.它采用了一种分治的策略,通常称其为分治法(Divide-and-ConquerMethod). 基本思想 先从数组中找出一个数作为基 ...
 - 快速排序算法Java版
		
网上关于快速排序的算法原理和算法实现都比较多,不过java是实现并不多,而且部分实现很难理解,和思路有点不搭调.所以整理了这篇文章.如果有不妥之处还请建议.首先先复习一些基础. 1.算法概念. ...
 - 快速排序算法Java实现
		
1) 通过一趟排序将要排序的数据分割成独立的两部分,其中一部分的所有数据都比另外一部分的所有数据都要小,然后再按此方法对这两部分数据分别进行快速排序,整个排序过程可以递归进行 示例: package ...
 - 一看就懂的Android APP开发入门教程
		
一看就懂的Android APP开发入门教程 作者: 字体:[增加 减小] 类型:转载 这篇文章主要介绍了Android APP开发入门教程,从SDK下载.开发环境搭建.代码编写.APP打包等步骤 ...
 - mysql取出现在的时间戳和时间时间戳转成人类看得懂的时间
		
mysql取出现在的时间戳和时间时间戳转成人类看得懂的时间,我们在mysql里面他封装了一个内置的时间戳转化的函数,比如我们现在的时间戳是:1458536709 ,"%Y-%m-%d&quo ...
 - 一看就懂的ReactJs入门教程(精华版)
		
一看就懂的ReactJs入门教程(精华版) 现在最热门的前端框架有AngularJS.React.Bootstrap等.自从接触了ReactJS,ReactJs的虚拟DOM(Virtual DOM)和 ...
 
随机推荐
- 方向键控制圆球运动(简易)(js)
			
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...
 - Node.js学习记录(一)--安装设置篇
			
安装Node window window上安装node可选择以下两种方式: 方式一:直接进入官网下载安装 进入node.js官网点击windows,选择.msi后缀的,根据自己的电脑选择对应的64位或 ...
 - ajax请求经典格式
			
$.ajax({ url: url, type: "POST", dataType: "json", data: data, success: funtion1 ...
 - Solr5.5高级应用(基于tomcat9)
			
一.配置solr 1.配置 注意:要是想放到其它路径下,可以修改此路径下的web.xml配置文件 修改内容如下: <!-- 将solrhome的绝对路径写入env-entry-value --& ...
 - celery任务进程关闭
			
方法1: ps auxww|grep 方法2: Ctrl+C 方法3: celery multi 管理 celery multi start w1 -A proj -l info celery mul ...
 - 自动化测试基础篇--Selenium cookie操作
			
摘自https://www.cnblogs.com/sanzangTst/p/8376471.html 在验证浏览器中cookie是否正确时,有时基于真实cookie的测试是无法通过白盒和集成测试进行 ...
 - 简单易懂的程序语言入门小册子(6):基于文本替换的解释器,引入continuation
			
当我写到这里的时候,我自己都吃了一惊. 环境.存储这些比较让人耳熟的还没讲到,continuation先出来了. 维基百科里对continuation的翻译是“延续性”. 这翻译看着总有些违和感而且那 ...
 - CharacterController平滑移动到某点
			
通常使用CharacterController控制玩家移动时,我们都会写以下代码: void Update() { var move = (moveTarget - transform.Positio ...
 - C++多线程同步技巧(一) --- 临界区
			
简介 C++中关于多线程的内容对于构建工程来说是至关重要的,C++本身也对关于多线程的操作提供了很好的支持.本章笔者就来介绍一下C++有关于多线程的重要知识点---临界区. 临界区的作用 线程就像是进 ...
 - 从 0 → 1,学习Linux该这么开始!
			
首先我们还是来普及以下概念,讲点虚的.现在是图形系统的天下,windows我们用了20多年.成功归功与它图形界面,你会点鼠标吗你会敲键盘吗?所以你会上网会聊天会玩游戏了.那么,0基础接触的Linux, ...