选择排序的双选优化

先整清楚选择排序

什么是选择排序?

矮子里面挑将军——双重循环遍历数组,每经过一轮比较,找到最小元素的下标,将其交换至首位,下次排序时它就不参与排序了,而是从剩余的数据中挑选最值。
请添加图片描述

手写一个选择排序

目标数组:{1, 0, 3, 8, 7, 2, 0, 5, 4, 6}
目标结果:顺序升序排序

public static void selectionSort() {
	int[] arr = {1, 0, 3, 8, 7, 2, 0, 5, 4, 6};
    int minIndex;
    for (int i = 0; i < arr.length - 1; i++) {
        minIndex = i;
        for (int j = i + 1; j < arr.length; j++) {
            if (arr[minIndex] > arr[j]) {
                // 记录最小值的下标
                minIndex = j;
            }
        }
        // 将最小元素交换至首位
        int temp = arr[i];
        arr[i] = arr[minIndex];
        arr[minIndex] = temp;
    }
    // 输出一下
	for(int i : arr) {
		System.err.print(i);
	}
}

选择排序和冒泡排序有什么区别?

回想下两种排序的原理图,还是有不一样的地方的
冒泡排序和选择排序虽然都是双for循环排序,时间复杂度也都是O(n^2)…巴拉巴拉…。但是!冒泡排序每轮选取最值时,这个比较的过程是在不断地交换的;选择排序每轮选取最值是通过一个变量记录最值索引,该轮比较结束后进行一次交换。可见选择排序很大程度上减少了交换次数。

你以为这就完了?这两种排序方式还有个极大的差别就是——冒泡排序是稳定的,选择排序是不稳定的


算法优化

最大值和最小值每轮比较只能得其一吗?

我们可以控制选择排序每轮比较选择出最大值还是者最小值,那么,我全都要呢?
本着这个目标,就上面的代码,稍作优化:

public static void selectionSort() {
    int[] arr = {1, 0, 3, 8, 7, 2, 0, 5, 4, 6};
    for (int i = 0; i < arr.length/2; i++) {
        System.out.print("第"+(i+1)+"次选最值——");
        int minIndex = i;
        int maxIndex = arr.length - 1 - i;
        for (int j = i + 1; j <= arr.length - 1 - i; j++) {
            if (arr[minIndex] > arr[j]) {
                // 记录最小值的下标
                minIndex = j;
            }
            if (arr[maxIndex] < arr[j]) {
                // 记录最大值的下标
                maxIndex = j;
            }
        }
        // 将最小元素交换至首位
        int temp = arr[i];
        arr[i] = arr[minIndex];
        arr[minIndex] = temp;
        System.out.print("最小值为:"+arr[i]);

		// 这步很关键!
        if (maxIndex == i){
            maxIndex = minIndex;
        }

        // 将最大元素交换至末位
        int tem = arr[arr.length - i - 1];
        arr[arr.length - i - 1] = arr[maxIndex];
        arr[maxIndex] = tem;
        System.out.println("最大值为:"+arr[arr.length - i - 1]);
    }
    // 输出一下
    for(int i : arr) {
        System.err.print(i);
    }
}

总结的话就不说了,外层循环次数少了一半,内层循环次数也少了一半,你懂的。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值