排序算法之2:选择排序 SelectSort

选择排序(Selection sort)是一种简单直观的排序算法。

它的工作原理如下。首先在未排序序列中找到最小(大)元素,存放到排序序列的起始位置,然后,再从剩余未排序元素中继续寻找最小(大)元素,然后放到已排序序列的末尾。以此类推,直到所有元素均排序完毕。

选择排序的主要优点与数据移动有关。如果某个元素位于正确的最终位置上,则它不会被移动。选择排序每次交换一对元素,它们当中至少有一个将被移到其最终位置上,因此对 n 个元素的表进行排序总共进行至多 n-1 次交换。在所有的完全依靠交换去移动元素的排序方法中,选择排序属于非常好的一种。

/**
 * 选择排序的思想其实和冒泡排序有点类似,都是在一次排序后把最小的元素放到最前面。
 * 但是过程不同,冒泡排序是通过相邻的比较和交换。而选择排序是通过对整体的选择。
 * 举个栗子,对 5,3,8,6,4 这个无序序列进行简单选择排序,
 * 首先要选择 5 以外的最小数来和 5 交换,也就是选择 3 和 5 交换,一次排序后就变成了 3,5,8,6,4.
 * 对剩下的序列一次进行选择和交换,最终就会得到一个有序序列。
 * 其实选择排序可以看成冒泡排序的优化,因为其目的相同,只是选择排序只有在确定了最小数的前提下才进行交换,大大减少了交换的次数。
 * 选择排序的时间复杂度为 O(n^2)
 *
 * @author wb
 */
public class SelectSort {
    public static void selectSort(int[] arr) {
        if (arr == null || arr.length == 0)
            return;

        int minIndex;
        for (int i = 0; i < arr.length - 1; i++) { //只需要比较n-1次
            minIndex = i;
            //从i+1开始比较,因为minIndex默认为i了,i就没必要比了。
            for (int j = i + 1; j < arr.length; j++) { 
                if (arr[j] < arr[minIndex]) {
                    minIndex = j;
                }
            }

            if (minIndex != i) { //如果minIndex不为i,说明找到了更小的值,交换之。
                swap(arr, i, minIndex);
            }
        }
    }

    public static void swap(int[] arr, int i, int j) {
        int temp = arr[i];
        arr[i] = arr[j];
        arr[j] = temp;
    }
}

白话分析:

  1. 从第一个元素循环遍历每一个元素,找到最小的元素,遍历一轮结束后和第一个元素进行交换,此时最小值处于数组前面;
  2. 重复步骤一,遍历剩余的序列;
  3. 每一轮完毕后剩余队列最小者都会被交换到最左侧;
  4. 最终得到一个有序队列;
排序算法之2:选择排序 SelectSort_第1张图片
选择排序.gif

你可能感兴趣的:(排序算法之2:选择排序 SelectSort)