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. ETL作业调度软件TASKCTL4.1集群部署

    熟悉TASKCTL4.1一段时间后,觉得它的调度逻辑什么的都还不错,但是感觉单机部署不太够用.想实现跨机调度作业,就要会TASKCTL的集群部署.下面就是我在网上找到的相关资料,非原创. 单机部署成功 ...

  2. springMVC 中几种获取request和response的方式

    1.最简单方式:参数 例如: @RequestMapping("/test") @ResponseBody public void saveTest(HttpServletRequ ...

  3. 闭包(Closure)基础分析

    闭包(Closure) 本文聚焦于回答2个问题: 在全局作用域中,如何读取函数内部的局部变量? 在全局作用域中,如何修改函数内部的局部变量? 变量作用域 JavaScript语言的作用域,一句话概括就 ...

  4. zabbix前台jsrpc注入

    zabbix是一个开源的企业级性能监控解决方案. 官方网站:http://www.zabbix.com zabbix的jsrpc的profileIdx2参数存在insert方式的SQL注入漏洞,攻击者 ...

  5. 正则表达式大全 --【Python举例】

    包含 :  纯文字.正负数,小数.正数.正整数.月份.天数.用户名.密码.车牌.传真.手机.邮件.ipv4私有地址.ipv4地址.ipv6地址.json_header.request_header 有 ...

  6. TypeScript基础数据类型

    Typescript与Javascript 二者的区别 作为前端开发的基础语言javascript已经深入人心,人人皆知.作为变成入门的最简单语言,Javascript语言以动态.弱类型语言而著名,学 ...

  7. The ways to kill Oracle session

    As we all known ,its the normal way  to use the SQL  'alter system kill 'sid,serial#'' to kill a ses ...

  8. 13. Roman to Integer【leetcode】

    Roman to Integer Given a roman numeral, convert it to an integer. Input is guaranteed to be within t ...

  9. python学习===判断两个日期的间距天数

    import datetime   d1 = datetime.date(2015,10,7) d2 = datetime.date(2015,8,15) print((d1-d2).days)

  10. [Mysql] 安装后启动不了

    Mysql安装后启动报错: ERROR 2002 (HY000): Can't connect to local MySQL server through socket '/var/lib/mysql ...