选择排序法

常用的选择排序方法有两种:直接选择排序堆排序
直接排序简单直观,但性能略差;
堆排序是一种较为高效的选择排序方法,但实现起来略微复杂。

直接选择排序

直接选择排序的思路很简单,它需要经过n-1趟比较。

  • 第1趟比较:程序将记录定位在第1个数据上,拿第1个数据依次和它后面每个数据进行比较,如果第1个数据大于后面某个数据,交换它们……依此类推。经过第1趟比较,这组数据中最小的数据被选出,它被排在第1位。
  • 第2趟比较:程序将记录定位在第2个数据上,拿第2个数据依次和它后面每个数据进行比较,如果第2个数据大于后面某个数据,交换它们……依此类推。经过第2趟比较,这组数据中第2小的数据被选出,它被排在第2位。
    ……
    按此规则一共进行n-1趟比较,这组数据中第n-1小(第2大)的数据被选出,被排在第n-1位(倒数第1位);剩下的就是最大的数据,它排在最后。

直接选择排序的优点是算法简单,容易实现。
直接选择排序的缺点是每趟只能确定一个元素,n个数组需要进行n-1趟比较。

封装的实体类

public class DataWrap implements Comparable<DataWrap>{
    int data;
    String flag;
    public DataWrap(int data, String flag) {
        this.data = data;
        this.flag = flag;
    }
    @Override
    public String toString() {
        return data+flag;
    }
    @Override
    public int compareTo(DataWrap dw) {
        return this.data > dw.data ? 1 : (this.data == dw.data ? 0 : -1);
    }
}

具体的算法与测试

public class SelectSort {

    public static void selectSort(DataWrap[] data) {
        System.out.println("开始排序");
        int arrayLength = data.length;

        //依次进行n-1趟比较,第i趟比较将第i大的值选出,放在i位置上
        for (int i = 0; i < arrayLength - 1; i++) {
            //第i个数据只需要和它后面的数据比较
            for (int j = i + 1; j < arrayLength; j++) {
                //如果第i位置的数据 大于 j位置上的数据,就交换他们
                if (data[i].compareTo(data[j]) > 0) {
                    DataWrap tmp = data[i];
                    data[i] = data[j];
                    data[j] = tmp;
                }
            }
            System.out.println("第 "+i+" 趟后:"+Arrays.toString(data));
        }
    }

    /**
     * 优化:
     *      在一趟的排序过程中,记录最小数据的位置。当本趟比较完成时,如果第i位于minIndex不相等时,才交换位置
     * @param data
     */
    public static void selectSort2(DataWrap[] data) {
        System.out.println("开始排序");
        int arrayLength = data.length;

        //依次进行n-1趟比较,第i趟比较将第i大的值选出,放在i位置上
        for (int i = 0; i < arrayLength - 1; i++) {
            //minIndex用于保留本趟比较中最小值的索引
            int minIndex = i;
            //第i个数据只需要和它后面的数据比较
            for (int j = i + 1; j < arrayLength; j++) {
                //如果第minIndex位置的数据 大于 j位置上的数据,将j的值赋给minIndex
                if (data[minIndex].compareTo(data[j]) > 0) {
                    minIndex = j;
                }
            }
            if (i != minIndex) {
                DataWrap tmp = data[i];
                data[i] = data[minIndex];
                data[minIndex] = tmp;
            }
            System.out.println("第 "+i+" 趟后:"+Arrays.toString(data));
        }
    }


    public static void main(String[] args) {
        DataWrap[] data = new DataWrap[]{
                new DataWrap(21, ""),
                new DataWrap(30, ""),
                new DataWrap(49, ""),
                new DataWrap(30, "*"),
                new DataWrap(16, ""),
                new DataWrap(9, "")
        };
        DataWrap[] data2 = Arrays.copyOf(data, data.length);

        System.out.println("排序前:"+Arrays.toString(data));
        selectSort(data);
        System.out.println("排序后:"+Arrays.toString(data));
        selectSort2(data2);
    }
}

总结

直接选择排序,时间效率为O(n2)
交换时,只需要一个附加程序单元用于交换,其空间效率为:O(1)
30与30* 排序后,位置发生了变化,所以排序是不稳定的

堆排序

堆的概念

假设有n个数据元素的序列k0,k1,…,kn-1,当且仅当满足如下关系时,可以将这组数据称为小顶堆(小根堆)。
ki <= k2i+1且ki <= k2i+2(其中i=0, 2,…,(n-1)/2)
或者,满足如下关系时,可以将这组数据称为大顶堆(大根堆)。
ki >= k2i+1且ki >=k2i+2(其中i=0, 2,…,(n-1)/2)
对于满足小顶堆的数据序列k0,k1,…,kn-1,如果将它们顺序排成一棵完全二叉树,则此树的特点是:树中所有节点的值都小于其左右子节点的值,此树的根节点的值必然最小。反之,对于满足大顶堆的数据序列k0,k1,…,kn-1,如果将它们顺序排成一棵完全二叉树,则此树的特点是:树中所有节点的值都大于其左右子节点的值,此树的根节点的值必然最大。
通过上面介绍不难发现一点,小顶堆的仁义子树也是小顶堆,大顶堆的任意子树还是大顶堆

例:判断数据序列
9,30,49,46,58,79是否为堆,将其转换为一个完全二叉树


小顶堆

判断数据序列:93,82,76,63,58,67,55是否为堆,将其转换为一个完全二叉树


大顶堆

堆排序的关键在于建堆,它按如下步骤完成排序。

  • 第1趟将索引0~n-1处的全部数据建大顶(或小顶)堆,就可以选择出这组数组中的最大(或最小)值。
  • 将上一步所建的大顶(或小顶)堆的根节点与这组数据的最后一个节点交换,就使得这组数据中最大(或最小)值排在最后。
  • 第2趟将索引0~n-2处的全部数据建大顶(或小顶)堆,就可以选择出这组数组中的最大(或最小)值。
    将上一步所建的大顶(或小顶)堆的根节点与这组数据的倒数第2个节点交换,就使得这组数据中最大(或最小)值排在倒数第2位。
    ……
  • 第k趟将索引0~n-k处的全部数据建大顶(或小顶)堆,就可以选择出这组数组中最大(或最小)值。
  • 将上一步所建的大顶(或小顶)堆的根节点与这组数据的倒数第k个节点交换,使得这组数据中最大(或最小)值排在倒数第k位。

通过上面介绍不难发现,堆排序的步骤就是重复执行以下2步。
(1)建堆;
(2)拿堆的根节点和最后一个节点交换。
由此可见,对于包含n个数据元素的数据组而言,堆排序需要经过n-1次建堆,每次建堆的作用就是选出该堆的最大值或最小值。因为堆排序的本质上依然是一种选择排序。

例如如下数据组:
9,79,46,30,58,49
建堆过程

  • 先将其转换为完全二叉树,转换得到的完全二叉,如下图


    将数据转换为完全二叉树
  • 完全二叉树的最后一个非叶子节点,也就是最后一个节点的父节点。最后一个节点的索引为数组长度-1,也就是len-1,那么最后一个非叶子节点的索引应该是为(len-2)/2。也就是从索引为2的节点开始,如果其子节点的值大于它本身的值,则把它和较大子节点进行交换,即将索引2处节点和索引5处的元素交换。交换后的结果如下图


    交换1次后的完全二叉树
  • 向前处理前一个节点,也就是处理索引为1的节点,此时79>30、79>58,因此无须交换。
  • 向前处理前一个节点,也就是处理索引为0的节点,此时 9<79、9<49,因此需要交换。应该拿索引0的节点和索引1的节点交换(9的两个子节点中,索引为1的节点的值较大),交换后的完全二叉树如下图


    交换2次后的完全二叉树
  • 如果某个节点和它的某个子节点交换后,该子节点又有子节点,系统还需再次对该子节点进行判断。例如,上图中索引 0的节点和索引 1的节点交换后,索引 1的节点还有子节点,因此程序必须再次保证索引1处节点的值大于等于其左、右子节点的值。因此还需要交换一次,交换后的大顶堆如下图


    大顶堆建立完成

具体算法

public class HeapSort {

    public static void heapSort(DataWrap[] data){
        System.out.println("开始排序");
        int arrryLength = data.length;
        //循环建堆,  第0个节点没有父节点的所以可以少比较一次
        for (int i = 0; i < arrryLength - 1; i++) {
            //建堆
            buildMaxdHeap(data, arrryLength - 1 - i);
            //交换堆顶和最后一个元素
            swap(data, 0, arrryLength - 1 - i);
            System.out.println(Arrays.toString(data));
        }
    }

    /**
     * 对data数据从0到lastIndex建大顶堆
     * @param data
     * @param lastIndex
     */
    private static void buildMaxdHeap(DataWrap[] data, int lastIndex){
        //从lastIndex处节点(最后一个节点)的父节点开始,依次循环其前面的各个父节点
        //说明: 不管lastIndex是左子树还是右子树, (lastIndex-1)/2 =(int)x,最后取整后,就是他的父节点
        for (int i = (lastIndex - 1) / 2; i >= 0; i--) {
            //k保存当前正在判断的节点
            int k=i;
            //如果当前k节点的子节点存在
            while (k * 2 + 1 <= lastIndex) {
                //k节点的左子节点的索引
                int biggerIndex = 2 * k + 1;
                //如果biggerIndex小于lastIndex,即biggerIndex+1
                //代表k节点的右子节点存在(因为lastIndex是最后一个节点, biggerIndex<lastIndex,2k+2是k节点的右子节点,那么2k+2<=lastIndex)
                if (biggerIndex < lastIndex) {
                    //如果右子节点的值较大
                    if (data[biggerIndex].compareTo(data[biggerIndex + 1]) < 0) {
                        //biggerIndex总是记录较大子节点的索引
                        biggerIndex++;
                    }
                }
                //如果k节点的值小于其他较大子节点的值
                if (data[k].compareTo(data[biggerIndex]) < 0) {
                    //交换它们
                    swap(data, k, biggerIndex);
                    //将biggerIndex赋给k,开始while循环的下一次循环
                    //重新保证k节点的值大于其左、右子节点的值(重新建立此节点的堆)
                    k = biggerIndex;
                }else{
                    break;
                }
            }
        }
    }

    /**
     * 交换data数组中i,j两个索引出的元素
     * @param data
     * @param i
     * @param j
     */
    private static void swap(DataWrap[] data, int i, int j){
        DataWrap tmp = data[i];
        data[i] = data[j];
        data[j] = tmp;
    }

    public static void main(String[] args) {
        DataWrap[] data = new DataWrap[]{
                new DataWrap(21, ""),
                new DataWrap(30, ""),
                new DataWrap(49, ""),
                new DataWrap(30, "*"),
                new DataWrap(21, "*"),
                new DataWrap(16, ""),
                new DataWrap(9, "")
        };

        System.out.println("排序前:" + Arrays.toString(data));
        heapSort(data);
        System.out.println("排序后:"+Arrays.toString(data));
    }
}

总结

  • 假设有n项数据,需要进行n-1次建堆,则其时间效率是O(log2n)
  • 堆排序也只需要一个附加程序单元用于交换,其空间效率为O(1)
  • 堆排序是不稳定的
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 219,869评论 6 508
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,716评论 3 396
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 166,223评论 0 357
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 59,047评论 1 295
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 68,089评论 6 395
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,839评论 1 308
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,516评论 3 420
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,410评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,920评论 1 319
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 38,052评论 3 340
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,179评论 1 352
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,868评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,522评论 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 32,070评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,186评论 1 272
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,487评论 3 375
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 45,162评论 2 356

推荐阅读更多精彩内容