package practice;

import edu.princeton.cs.algs4.*;

public class TestMain {
public static void main(String[] args) {
int[] a = new int[20];
for (int i = 0; i < a.length; i++) {
int temp = (int)(StdRandom.uniform(1, 10));
a[i] = temp;
} for (int i : a) {
System.out.print(i+" ");
}
System.out.println();
ToSort.sort3Way(a);
for (int i : a) {
System.out.print(i+" ");
}
}
} class ToSort{
private static int[] anx;
/*
* 我第一次写出来的快速排序,容易理解,但需要一倍的额外空间,也不快.....
*/
public static void mySort(int[] a) {
anx = new int[a.length];
myQuickSort(a, 0, a.length - 1);
}
public static void myQuickSort(int[] a,int lo,int hi) {
if (lo >= hi) return;
int m = lo;
int n = hi;
int mid = a[lo];
for (int i = lo + 1; i <= hi; i++) {
if (a[i] <= mid) anx[m++] = a[i];
else if (a[i] > mid) anx[n--] = a[i];
}
//此处往后m,n相等
anx[m] = mid;
for (int i = lo; i <= hi; i++) a[i] = anx[i];
myQuickSort(a, lo, m - 1);
myQuickSort(a, n + 1, hi);
}
/*
* 快速排序 时间复杂度O(NlgN)
*/
public static void sort(int[] a) {
quickSort(a, 0, a.length - 1);
}
public static void quickSort(int[] a,int lo,int hi) {
if (lo >= hi) return; //如果分到只剩一个元素,或没有元素,则返回
int n = partition(a, lo, hi); //将数组切分,取一个中值,分成比它小的一部分和比它大的一部分
quickSort(a, lo, n - 1); //将两部分分别快速排序
quickSort(a, n + 1, hi);
}
public static int partition(int[] a,int lo,int hi) {
int mid = a[lo]; //把第一个元素设为中值,先放在最左边不管
int m = lo;
int n = hi + 1;
while (true) {
while (a[++m] < mid) if(m == hi) break; //从左开始找出一个比mid大的元素
while (a[--n] > mid)/*if(n == lo) break; 这句没用,但可以让人更容易理解程序*/;//从右开始找出一个比mid小的元素
if (m >= n) break; //找完了
exch(a, m, n);
}
exch(a, n, lo); //把最左端中值和"比它小的值中最右端的数"换位置,它就到最中间了
return n;
}
/*
* 三向切分的快速排序,适用于有大量重复元素的数组
*/
public static void sort3Way(int[] a) {
quickSort(a, 0, a.length - 1);
}
public static void quickSort3Way(int[] a,int lo,int hi) {
if (lo >= hi) return;
int mid = a[lo];
int lt = lo + 1;
int gt = hi;
int i = lo + 1;
while (i <= gt) { //将比中值小的放在最左端,比中值大的放在最右端,和中值相等的放在中间
if (a[i] < mid) exch(a, lt++, i++); //换过来的值还是与中值相等的值,所以不用动
else if (a[i] > mid) exch(a, gt--, i); //换过来的值不知道是什么值,所以还要处理
else if (a[i] == mid) i++; //相等就过
}
exch(a, lo, lt - 1); //把最左端中值和"比它小的值中最右端的数"换位置
quickSort3Way(a, lo, lt - 2); //调用自己处理剩下的两部分
quickSort3Way(a, gt + 1, hi);
}
/*
* 交换a[i]与a[j]的值
*/
private static void exch(int[] a, int i, int j) {
int t = a[i];
a[i] = a[j];
a[j] = t;
}
}

快速排序示意图(图片来自《算法(第四版官网)》)

三向切分的快速排序示意图(图片来自《算法(第四版官网)》)

快速排序Java实现的更多相关文章

  1. 快速排序 Java实现的快速排序

    快速排序  Java实现的快速排序: package xc; import java.util.Arrays; import java.util.Random; /** * * @author dax ...

  2. 基本排序算法——快速排序java实现

    简单的快速排序算法,我竟然花费了如此多的时间来写作,好好学习. /** * */ package basic.sort; import java.util.Arrays; import java.ut ...

  3. 排序算法----快速排序java

    快速排序是对冒泡排序的一种改进,平均时间复杂度是O(nlogn) import java.util.Arrays; import java.util.Scanner; public class tes ...

  4. 快速排序 java详解

    1.快速排序简介: 快速排序由C. A. R. Hoare在1962年提出.它的基本思想是:通过一趟排序将要排序的数据分割成独立的两部分,其中一部分的所有数据都比另外一部分的所有数据都要小,然后再按此 ...

  5. ADV-297 快速排序 java

    问题描述 用递归来实现快速排序(quick sort)算法.快速排序算法的基本思路是:假设要对一个数组a进行排序,且a[0] = x.首先对数组中的元素进行调整,使x放在正确的位置上.同时,所有比x小 ...

  6. 快速排序-java

    排序-快速排序 基本思想: 将数据划分为两部分,左边的所有元素都小于右边的所有元素:然后,对左右两边进行快速排序. 划分方法: 选定一个参考点(中间元素),所有元素与之相比较,小的放左边,大的放右边. ...

  7. 数组排序-冒泡排序-选择排序-插入排序-希尔排序-快速排序-Java实现

    这五种排序算法难度依次增加. 冒泡排序: 第一次将数组相邻两个元素依次比较,然后将大的元素往后移,像冒泡一样,最终最大的元素被移到数组的最末尾. 第二次将数组的前n-1个元素取出,然后相邻两个元素依次 ...

  8. 排序算法之快速排序(java实现)

    package com.javaTest300; public class Test039 { public static void main(String[] args) {// 快速排序 int ...

  9. 快速排序java

    快速排序(Quicksort)是对冒泡排序的一种改进.它是先在数组中找到一个关键数,第一趟排序将比关键数小的放在它的左边,比关键数大的放在它的右边.当第一趟排序结束后,再依次递归将左边和右边的进行排序 ...

随机推荐

  1. 小白浅论JAVA数组中“for加强版”

    代码: /*String[] s=new String[]{"sdsfd","fgd","sdf"}; for(String a:s) Sy ...

  2. Python 获取当前路径的方法

    Python2.7 中获取路径的各种方法 sys.path 模块搜索路径的字符串列表.由环境变量PYTHONPATH初始化得到. sys.path[0]是调用Python解释器的当前脚本所在的目录. ...

  3. gdb命令中查看地址之x命令

    可以使用examine命令(简写是x)来查看内存地址中的值.x命令的语法如下所示: x/<n/f/u> <addr> n.f.u是可选的参数. n是一个正整数,表示需要显示的内 ...

  4. dubbo源码分析(二):超时原理以及应用场景

    dubbo超时原理以及应用场景 *:first-child { margin-top: 0 !important; } body > *:last-child { margin-bottom: ...

  5. 刚装的系统C盘占空间特别大怎么办?关闭win7的系统还原和调整虚拟内存

    刚装的win7系统,C盘占空间特别大,清理垃圾,清理碎片之后还是特别大,检查一下,系统才20G,C盘却占用了40G,其实是没有关闭系统还原,再就是没有调整虚拟内存.win7旗舰版(其他版本也可能这样, ...

  6. vue+mockjs 模拟数据,实现前后端分离开发

    在项目中尝试了mockjs,mock数据,实现前后端分离开发. 关于mockjs,官网描述的是 1.前后端分离 2.不需要修改既有代码,就可以拦截 Ajax 请求,返回模拟的响应数据. 3.数据类型丰 ...

  7. JavaScript高程--<script>标签

    <script>标签 在HTML5中script主要有以下几个属性:async,defer,charset,src,type, async(可选): 关键词:异步脚本,外部文件,立即下载: ...

  8. Comparable与Comparator,java中的排序与比较

    1:比较和排序的概念 比较:两个实体类之间按>,=,<进行比较. 排序:在集合类中,对集合类中的实体进行排序.排序基于的算法基于实体类提供的比较函数. 基本型别都提供了默认的比较算法,如s ...

  9. knockoutjs模板实现树形结构列表

    数据结构 /*数据*/ var ko_vue_data=[ { name: "总能耗", number:"0", energyone: 14410, energ ...

  10. null == undefined ?

    最近在看<JavaScript高级程序设计>一书,书中讲到相等操作符(==)时说,要比较相等性之前,不能将 null 和 undefined 转换成其他任何值,但要记住 null == u ...