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. JS浏览器对象:window对象、History、Location对象、Screen对象

    一.JS浏览器对象-window 1.window对象 window对象是BOM的核心,window对象指当前的浏览器窗口 所有JavaScript全局对象.函数以及变量均自动成为window对象的成 ...

  2. linux的文件权限与目录配置<----->第二章

    1.Linux文件属性    [   1  ]                  [   2  ] [    3    ]  [    4    ]  [ 5 ]             [ 6 ] ...

  3. Verilog HDL常用综合语法

    前面已经记录了一些组成Verilog的基本组成,可以用这些基本组成来构成表达式.这一节,就来记录一下把这些表达式构成一个文件的各种行为描述语句. ①这里用Verilog基本要素进行的行为描述主要是针对 ...

  4. Tencent分布式开源框架Pebble

    最近研究了T分布式开源框架Pebble,基本上想要的基础组件都有了,不过文档很糟糕,这也是T特色,只管开源不管维护:1.支持rpc/reverse_rpc2.支持http驱动3.支持tcp驱动(基于z ...

  5. Vue项目搭建完整剖析全过程

    Vue项目搭建完整剖析全过程 项目源码地址:https://github.com/ballyalex 有帮助的话就加个星星呗~!  项目技术栈:vue+webpack+bower+sass+axios ...

  6. 阿里消息队列中间件 RocketMQ源码解析:Message发送&接收

  7. NYOJ--STL--擅长排列的小明(强大的string :: iterator 和next_permutation)

    NYOJ--STL--擅长排列的小明 #include <iostream> #include <string> #include <algorithm> usin ...

  8. 移动Web学习笔记(第1天)-bootstrap框架的使用

    移动web前言 移动web : 移动端手机浏览器或者微信里面浏览的网页 移动APP : 手机上需要下载安装的应用程序 1. 移动web介绍 1.1 3 天 响应式开发 一套代码运行多个终端 优点:开发 ...

  9. 微信小程序(有始有终,全部代码)开发--- 新增【录音】以及UI改进

    开篇语 寒假发了一篇练手文章,不出意外地火了: <简年15: 微信小程序(有始有终,全部代码)开发---跑步App+音乐播放器 > 后来又发了BUG修复的版本,出乎意料的火了: 简年18: ...

  10. yhTriangle_LinkQueue(队列实现杨辉三角)

    #include"LinkQueue.h" void yhTriangle(int n) { LinkQueue<int> A; int s,t; A.Inqueue( ...