排序

初级排序算法

1.1 选择排序

思想

找到数组中最小的元素,然后和数组中第一个元素进行交换。然后找到剩下数中最小的元素,和数组的第二个元素进行交换。如此往复,直到将整个数组排序。

特点
  • 运行时间和输入无关
  • 数据移动是最少的

1.2 插入排序

思想

从左向右,将每一个元素移动到左边正确的位置。左边的数组始终是有序的。

特点
  • 运行时间取决于输入元素的初始顺序
  • 对于部分有序的数组比较高效

1.3 希尔排序

思想

由于插入排序一次只能移动1位,希尔排序每次移动h位。数组中任意间隔为h的元素都是有序的。

特点
  • 运行时间不能定量
package edu.princeton.cs.algs4.chapter2;

/**
 * 排序方法
 * Created by TXH-pc on 2017/3/4.
 */
public class Sort {

    /**
     * 选择排序
     */
    public static void selectionSort(Comparable[] a) {
        int N = a.length;
        for (int i = 0; i < N; i++) {
            int min = i;
            for (int j = i+1; j< N; j++) {
                if(less(a[j], a[min])) {
                    min = j;
                }
            }
            exch(a, i, min);
        }
    }

    /**
     * 插入排序
     */
    public static void insertionSort(Comparable[] a) {
        int N = a.length;
        for (int i = 1; i < N; i++) {
            //   终止条件是到达数组的最左端或者前面的元素已经比当前元素小
            for(int j = i; j > 0 && less(a[j], a[j - 1]) ; j--) {
                exch(a, j - 1, j);
            }
        }
    }

    /**
     * 希尔排序
     * @param a
     */
    public static void shell(Comparable[] a) {
        int N = a.length;
        int h = 1;
        while (h < N) {
            h = 3 * h + 1;
        }
        while (h >= 1) {
            for (int i = h; i < N; i++) {
                for (int j = i; j >= h && less(a[j], a[j - h]); j-=h) {
                    exch(a, j - h, j);
                }
            }
            h = h / 3;
        }

    }

    private static boolean less (Comparable v, Comparable w) {
        return v.compareTo(w) < 0;
    }

    private static void exch(Comparable[] a, int i, int j) {
        Comparable temp = a[i];
        a[i] = a[j];
        a[j] = temp;
    }

    private static void show(Comparable[] a) {
        for (int i = 0; i < a.length; i++) {
            System.out.print(a[i] + " ");
        }
        System.out.println();
    }

    public static void main(String[] args) {
        String[] a = {"t", "h", "i", "s", "i", "s", "t", "e", "s", "t"};
        Sort.selectionSort(a);
        Sort.insertionSort(a);
        Sort.shell(a);
        Sort.show(a);
    }
}

归并排序

基本思想

将两个有序的数组合并成一个更大的有序数组,有两种方法来实现归并排序:自顶而下的归并排序(分治法)和自底向上的归并排序。

2.1 分治法

  1. 新建一个临时数组(成员变量),merge的时候用来存放要排序的元素。
  2. 使用递归,将原来的数组分成两半,进行排序,并将排好序的数组复制到临时数组当中,最后将排好序的数组归并成一个数组。
  3. 在归并的过程中,分别用两个指针指向两个已经排好序的数组的初始位置,通过比较,将较小的元素放到合并数组。
分治法排序示例

Note:

  1. sort方法的作用在于安排多次merge方法调用的正确顺序
  2. 对于小规模的数组,可能插入排序比归并排序更快
  3. 对于长度为N的任意数组,自顶向下的归并排序至少需要1/2NlgN~NlgN次比较


    归并排序时间与NlgN成正比的证明

代码实现

public class MergeSort {

    private static Comparable[] aux; // 用于辅助的排序数组

    public static void sort(Comparable[] a) {
        aux = new Comparable[a.length]; // 初始化辅助数组
        // 使用递归,进行排序
        sort(a, 0, a.length - 1);
    }

    private static void sort(Comparable[] a, int lo, int hi) {
        // 递归终止条件
        if (hi <= lo)
            return;

        int mid = (lo + hi) / 2;
        sort(a, lo, mid); //排序左半边
        sort(a, mid + 1, hi); //排序右半边
        merge(a, lo, mid, hi); //合并
    }

    private static void merge(Comparable[] a, int lo, int mid, int hi) {
        int i = lo, j = mid + 1;

        for (int k = lo; k <= hi; k++) {
            aux[k] = a[k]; // 将要排序的数组移动到辅助数组
        }
        // 按大小进行比较归并,如果一边比较完,直接把另一边的元素复制回去
        for (int k = lo; k <= hi; k++) {
            if (i > mid)
                a[k] = aux[j++];
            else if (j > hi)
                a[k] = aux[i++];
            else if (aux[i].compareTo(aux[j]) < 0)
                a[k] = aux[i++];
            else
                a[k] = aux[j++];
        }
    }

    private static void show(Comparable[] a) {
        for (int i = 0; i < a.length; i++) {
            System.out.print(a[i] + " ");
        }
        System.out.println();
    }

    public static void main(String[] args) {
        Integer[] a = {1, 2, 5, 6, 3, 8, 6, 9};
        MergeSort.sort(a);
        MergeSort.show(a);
    }

}

2.2 自底向上

  1. 按顺序对数组中每两个元素进行排序
  2. 按顺序,每次对2的n次方个元素进行排序,直到所有的元素排序完成

代码实现

public class MergeSort {

    private static Comparable[] aux; // 用于辅助的排序数组

    /**
     * 自底向上的实现
     * @param a
     */
    public static void sortBU(Comparable[] a) {
        aux = new Comparable[a.length];
        int N = a.length;

        for (int sz = 1; sz < N; sz = sz + sz) { // sz为每次排序元素(子数组)的数量
            for (int lo = 0; lo < N - sz; lo += sz) {
                // 每次merge两个子数组,最后一个子数组的长度可能小于sz
                merge(a, lo, lo + sz - 1, Math.min(lo + sz + sz - 1, N - 1));
            }
        }
    }

    private static void merge(Comparable[] a, int lo, int mid, int hi) {
        int i = lo, j = mid + 1;

        for (int k = lo; k <= hi; k++) {
            aux[k] = a[k]; // 将要排序的数组移动到辅助数组
        }
        // 按大小进行比较归并,如果一边比较完,直接把另一边的元素复制回去
        for (int k = lo; k <= hi; k++) {
            if (i > mid)
                a[k] = aux[j++];
            else if (j > hi)
                a[k] = aux[i++];
            else if (aux[i].compareTo(aux[j]) < 0)
                a[k] = aux[i++];
            else
                a[k] = aux[j++];
        }
    }

    private static void show(Comparable[] a) {
        for (int i = 0; i < a.length; i++) {
            System.out.print(a[i] + " ");
        }
        System.out.println();
    }

    public static void main(String[] args) {
        Integer[] a = {1, 2, 5, 6, 3, 8, 6, 9};
        MergeSort.sortBU(a);
        MergeSort.show(a);
    }
}

排序算法的复杂度

没有任何基于比较的算法能保证少于lg(N!) ~ NlgN次比较将长度为N的数组排序

参考链接

http://rason.me/2016/06/29/MergeSort/
http://blog.acbingo.cn/2015/11/26/%E7%AE%97%E6%B3%95%E5%AD%A6%E4%B9%A0%E7%AC%94%E8%AE%B0-8-%E5%90%88%E5%B9%B6%E6%8E%92%E5%BA%8F/
http://www.cnblogs.com/yangecnu/p/Introduce-Merge-Sort.html

快速排序

基本思想

快速排序是先将切分元素放到合适的位置,切分元素左边的值总是小于切分元素,切分元素右边的值总是大于切分元素。通过递归,将左右两边继续切分,直到排序完成。

切分目的

  1. 对于某个j,a[j]已经确定
  2. a[lo]到a[j - 1]中的所有元素都不大于a[j]
  3. a[j + 1]到a[hi]中的所有元素都不小于a[j]
快速排序切分示意图

切分实现步骤

  1. 用两个指针(lo和hi),分别指向数组的头和尾
  2. 将指针指向的元素与切分元素(一般是数组的第一个元素)进行比较,如果a[lo]小于待切分的元素,则lo后移;如果a[hi]大于待切分的元素,则hi前移。
  3. 当lo指向一个大于切分元素的值时,停止移动,如果hi指向一个小于切分元素的值时,停止移动。此时,交换lo和hi所指向的值。
  4. 如此往复,直到hi<=lo,此时,交换切分元素和hi指向的值,同时返回hi

性能分析

  1. 将长度为N的无重复数组排序,快速排序平均需要~2NlnN次比较
  2. 快速排序最多需要N*N/2次比较,但是随机打乱数组能预防这种情况
  3. 当重复元素很多时,使用三项切分快速排序能将算法的时间复杂度降到N

代码实现

package edu.princeton.cs.algs4.chapter2;


/**
 * 快速排序
 * Created by TXH-pc on 2017/3/5.
 */
public class QuickSort {

    public static void sort(Comparable[] a) {
        // 打乱数组 略过
        sort(a, 0, a.length - 1);
        //quick3WaySort(a, 0, a.length - 1);
    }

    private static void sort(Comparable[] a, int lo, int hi) {
        // 递归停止条件
        if (hi <= lo)
            return;

        int j = partition(a, lo, hi);
        sort(a, lo, j - 1);
        sort(a, j + 1, hi);
    }


    private static int partition(Comparable[] a, int lo, int hi) {
        int i = lo, j = hi + 1; // 左右扫描指针
        Comparable v = a[lo];
        while (true) {
            while(less(a[++i], v)) {
                if (i == hi)
                    break;
            }
            while(less(v, a[--j])) {
                if (j == lo)
                    break;
            }
            if (j <= i)
                break;
            exch(a, i, j);
        }
        exch(a, lo, j); // 将v放入正确的位置
        return j;
    }

    /**
     * 三项切分快速排序
     * @param a
     * @param lo
     * @param hi
     */
    private static void quick3WaySort(Comparable[] a, int lo, int hi) {
        // 递归的终止条件
        if (hi <= lo)
            return;

        int lt = lo, i = lo + 1, gt = hi;
        Comparable v = a[lo];
        while (i <= gt) {
            int cmp = a[i].compareTo(v);
            if (cmp < 0)
                exch(a, lt++, i++);
            else if (cmp > 0)
                exch(a, i, gt--);
            else
                i++;
        }
        quick3WaySort(a, lo, lt - 1);
        quick3WaySort(a, gt + 1, hi);
    }

    private static void exch(Comparable[] a, int i, int j) {
        Comparable temp = a[i];
        a[i] = a[j];
        a[j] = temp;
    }

    private static boolean less (Comparable v, Comparable w) {
        return v.compareTo(w) < 0;
    }

    private static void show(Comparable[] a) {
        for (int i = 0; i < a.length; i++) {
            System.out.print(a[i] + " ");
        }
        System.out.println();
    }

    public static void main(String[] args) {
        String[] a = {"R", "B", "W", "W", "R", "W", "B", "R", "R", "W", "B", "R"};
        QuickSort.sort(a);
        show(a);
    }
}

优先队列

简介及比较

每次出来的都是最小值/最大值,可以用来查找海量数据中最大/最小的N个数
栈 :先进后出
队列 :先进先出
随机队列 : 随机出
优先队列:每次出来的是最大值或最小值

API介绍

public class MaxPQ<Key extends Comparable<Key>>
{
            MaxPQ()            // 创建一个优先队列
            MaxPQ(int max)     // 创建一个初始容量为max的优先队列
            MaxPQ(Key[] a)     // 用a[]中的元素创建一个优先队列
    void    insert(Key v)      // 向优先队列中插入一个元素
    Key     max()              // 返回最大是元素
    Key     delMax()           // 删除并返回一个最大元素
    boolean isEmpty()          // 优先队列是否为空
    int     size()             // 优先队列中的元素数量
}

优先队列实现的几种思路

  • 数组实现(无序)
    用一个数组存储数据,insert()时依次向数组中添加数据,delMax()时找出最大的元素,和边界元素进行交换,然后删除它。同时可以加入调整数组的大小的代码。
package edu.princeton.cs.algs4.chapter2;

/**
 * 无序数组实现的优先队列
 * Created by TXH-pc on 2017/3/5.
 */
public class UnorderedMaxPQ<Key extends Comparable<Key>> {
    private Key[] pq;
    private int N;

    public UnorderedMaxPQ() {}           // 创建一个优先队列

    // 创建一个初始容量为max的优先队列
    public UnorderedMaxPQ(int max) {
        pq = (Key[]) new Comparable[max];
    }

    // 向优先队列中插入一个元素
    public void insert(Key v) {
        pq[N++] = v;
    }

    // 删除并返回一个最大元素
    public Key delMax() {
        int max = 0;
        for (int i = 0; i< N; i++) {
            if (less(max, i))
                max = i;
        }
        exch(max, N - 1);
        return pq[--N];
    }

    private void exch(int max, int i) {
        Key temp = pq[max];
        pq[max] = pq[i];
        pq[i] = temp;
    }

    private boolean less(int max, int i) {
        if(pq[max].compareTo(pq[i]) < 0)
            return true;
        else
            return false;
    }

    // 优先队列是否为空
    public boolean isEmpty() {
        return N == 0;
    }

    // 优先队列中的元素数量
    public int size() {
        return N;
    }

    public static void main(String[] args) {
        UnorderedMaxPQ<String> pq = new UnorderedMaxPQ<String>(3);
        pq.insert("f");
        pq.insert("z");
        pq.insert("j");
        System.out.println(pq.delMax());
    }
}
  • 数组实现(有序)
    在插入数组的时候,将较大的元素向右移动一格以保证数组有序,这样最大的元素总在数组的一边。
  • 链表表示法(有序&无序)
  • 堆实现
优先队列的各种实现在最坏情况下运行时间的增长量级

二叉堆实现优先队列

  • 二叉堆的每个节点都大于等于它的两个子节点
  • 使用数组表示二叉堆的时候,索引为0的位置不放置元素。那么,位置k的节点的父节点位置为k/2,两个子节点的位置为2k和2k+1
  • 由下至上的堆有序化(上浮):当某种原因导致一个节点比它的父节点更大的时候,需要交换它和它父节点的位置,循环往复直到交换后的父节点比当前节点大
上浮过程
  • 由上至下的堆有序化(下沉):当某种原因导致一个节点比它的两个子节点更小的时候,需要将它和它的两个子节点当中较小的进行交换来恢复堆
  • 插入元素操作:将要插入的元素放置在数组的末尾,增加堆的大小然后通过上浮操作使堆恢复到有序的状态


    下沉过程
  • 删除最大元素:从数组顶端去掉最大元素,然后将数组的最后一个元素放置到顶端,减小堆的大小然后通过下沉操作恢复堆的有序状态

基于二叉堆的优先队列的实现

package edu.princeton.cs.algs4.chapter2;

/**
 * 基于堆的优先队列
 * Created by TXH-pc on 2017/3/5.
 */
public class MaxPQ<Key extends Comparable<Key>> {
    private Key[] pq;
    private int N = 0; // 元素存储在[1...N]当中,pq[0]不使用

    public MaxPQ(int maxN) {
        pq = (Key[]) new Comparable[maxN + 1];
    }

    public boolean isEmpty() {
        return N == 0;
    }

    public int size() {
        return N;
    }

    public void insert(Key v) {
        pq[++N] = v;
        swim(N);
    }

    public Key delMax() {
        Key max = pq[1];
        exch(1, N--);
        pq[N+1] = null;
        sink(1);
        return max;
    }

    private void swim(int k) {
        while (k > 1 && less(k/2, k)) {
            exch(k / 2, k);
            k = k / 2;
        }
    }

    private void sink(int k) {
        // 停止条件应该是到达底端或者子节点都比他小
        while (2*k <= N) {
            int j = 2*k;
            if (j < N && less(j, j+1))
                j++;
            if (less(k, j))
                break;
            exch(k, j);
            k = j;
        }
    }

    private boolean less(int i, int j) {
        return pq[i].compareTo(pq[j]) < 0;
    }

    private void exch(int i, int j) {
        Key temp = pq[i];
        pq[i] = pq[j];
        pq[j] = temp;
    }

    public static void main(String[] args) {
        MaxPQ<String> pq = new MaxPQ<String>(3);
        pq.insert("f");
        pq.insert("a");
        pq.insert("j");
        System.out.println(pq.delMax());
    }
}

堆排序

堆排序分两个阶段:

  1. 构造堆结构

通过从右至左用sink()构造子堆,只需要扫描数组中的一半的元素

  1. 下沉排序

将堆中最大的元素和最后一个元素交换,然后放入堆缩小后空出的位置。然后使用sink()方法使堆有序

package edu.princeton.cs.algs4.chapter2;

/**
 * 堆排序
 * Created by TXH-pc on 2017/3/5.
 */
public class HeapSort {

    public static void sort(Comparable[] a) {
        int N = a.length;
        // 堆的构造
        for (int k = N / 2; k >= 1; k--) {
            sink(a, k, N);
        }
        // 下沉排序
        while (N > 1) {
            exch(a, 1, N--);
            sink(a, 1, N);
        }
    }

    private static void sink(Comparable[] a, int k, int N) {
        while (2*k <= N) {
            int j = 2*k;
            if (j < N && less(a, j, j+1))
                j++;
            if (less(a, j, k))
                break;
            exch(a, j, k);
            k = j;
        }

    }

    // following two functions should convert from 1-base to 0 base
    private static void exch(Comparable[] a, int j, int k) {
        Comparable temp = a[j - 1];
        a[j - 1] = a[k - 1];
        a[k - 1] = temp;
    }

    private static boolean less(Comparable[] a, int i, int j) {
        return a[i - 1].compareTo(a[j - 1]) < 0;
    }

    private static void show(Comparable[] a) {
        for (int i = 0; i < a.length; i++) {
            System.out.print(a[i] + " ");
        }
        System.out.println();
    }

    public static void main(String[] args) {
        String[] a = {"t", "h", "i", "s", "i", "s", "t", "e", "s", "t"};
        HeapSort.sort(a);
        HeapSort.show(a);
    }
}

算法

  • Stable means if the two elements have the same key, they remain in the same order or positions. But that is not the case for Heap sort.
算法比较
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,928评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,192评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,468评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,186评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,295评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,374评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,403评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,186评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,610评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,906评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,075评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,755评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,393评论 3 320
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,079评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,313评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,934评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,963评论 2 351

推荐阅读更多精彩内容

  • 最近在读< >时,了解到了很多常用的排序算法,故写一篇读书笔记记录下这些排序算法的思路和实现. 冒泡排序 冒泡排序...
    SylvanasSun阅读 685评论 0 0
  • 四. 走向世界之巅——快速排序 你可能会以为归并排序是最强的算法了,其实不然。回想一下,归并的时间效率虽然高,但空...
    Leesper阅读 1,652评论 9 7
  • 排序的基本概念 在计算机程序开发过程中,经常需要一组数据元素(或记录)按某个关键字进行排序,排序完成的序列可用于快...
    Jack921阅读 1,423评论 1 4
  • 一. 写在前面 要学习算法,“排序”是一个回避不了的重要话题,在分析完并查集算法和常用数据结构之后,今天我们终于可...
    Leesper阅读 2,528评论 0 40
  • 关键词3 回忆、伤感、遇见 一 回到家,已经快凌晨一点了。 我把台灯打开,从书桌的抽屉里拿出纸盒子,里面只有被我撕...
    阿畔阅读 712评论 11 13