:-: 14.3 选择排序
* * * * *
**基本概念:**
它是一种简单直观的排序算法。它的工作原理如下。首先在未排序序列中找到最小(大)元素,存放到排序序列的起始位置,然后,再从剩余未排序元素中继续寻找最小(大)元素,然后放到已排序序列的末尾。以此类推,直到所有元素均排序完毕。
它的主要优点与数据移动有关。如果某个元素位于正确的最终位置上,则它不会被移动。选择排序每次交换一对元素,它们当中至少有一个将被移到其最终位置上,因此对n个元素的表进行排序总共进行至多n-1次交换。在所有的完全依靠交换去移动元素的排序方法中,选择排序属于非常好的一种。
**算法的运作:**
1、首先在未排序序列中找到最小(大)元素,若最后所得最小(大)值的位置和当前对比的元素位置不同,就交换存放到排序序列的当前对比的元素位置,若相同直接跳过从下一个开始。
2、再从剩余未排序元素中继续寻找最小(大)元素的位置。
3、以此类推,重复以上步骤,直到所有元素均排序完毕。
**图片示例:**
:-: ![](https://box.kancloud.cn/1416d6bf9e2daf8b231abe99a1bf5647_811x248.gif)
**代码实现:**
第一种方式:
~~~
public void selectionSort(int nums[]) {
int min = 0, sign = 0;
for (int i = 0; i < nums.length - 1; i++) {
min = i;
for (int j = min + 1; j < nums.length; j++) {
if (nums[min] > nums[j]) {
min = j;
}
}
if (min != i) {
sign = nums[min];
nums[min] = nums[i];
nums[i] = sign;
}
}
}
~~~
第二种方式:我称为鸡尾酒选择排序,一轮选两个,就是从两边向中间,逐步有序,左边是选择出的最小值,右边是选择出的最大值,直到左右交合。
**图片示例:**
:-: ![](https://box.kancloud.cn/9a387138d8e28b7ea3fc99339e7ee7f5_810x502.gif)
~~~
public void selectionSort(int nums[]) {
int left = 0, right = nums.length - 1, j, min, max, temp = 0;
while (left < right) {
min = left;
for (j = left + 1; j <= right; j++) {
if (nums[min] > nums[j]) {
min = j;
}
}
if (min != left) {
temp = nums[min];
nums[min] = nums[left];
nums[left] = temp;
}
left++;
max = right;
for (j = right - 1; j >= left; j--) {
if (nums[max] < nums[j]) {
max = j;
}
}
if (max != right) {
temp = nums[max];
nums[max] = nums[right];
nums[right] = temp;
}
right--;
}
}
~~~
**复杂度分析:**
时间复杂度:平均:O(n^2)、最坏:O(n^2)、最优:O(n^2)。
空间复杂度:最坏:O(n), 最优:O(1)。
**若有理解错误的、写错的地方、更好的思路,方法,望各位读者评论或者发邮件指正或指点我,不胜感激。**
**本文主要参考以下文章:**
[维基百科——选择排序](https://zh.wikipedia.org/wiki/%E9%80%89%E6%8B%A9%E6%8E%92%E5%BA%8F)
[JS中常见排序算法(示例图片来源)](https://github.com/Sir814/Sort/blob/master/02.%E9%80%89%E6%8B%A9%E6%8E%92%E5%BA%8F.gif)
- 前言
- 第一部分 初级入门算法
- 第一章 数组
- 1.1 删除排序数组中的重复项
- 1.2 删除排序数组中的重复项 II
- 1.3 买卖股票的最佳时机
- 1.4 买卖股票的最佳时机 II
- 1.5 移动零
- 1.6 区间子数组个数
- 1.7 搜索插入位置
- 1.8 合并两个有序数组
- 1.9 两个数组的交集
- 第二章 哈希表
- 2.1 两数之和
- 2.2 错误的集合
- 2.3 翻转卡片游戏
- 2.4 有效的字母异位词
- 第三章 链表
- 第四章 数学
- 4.1 加一
- 4.2 反转整数
- 4.3 排列硬币
- 4.4 完全平方数
- 4.5 消除游戏
- 第五章 双指针
- 第六章 字符串
- 6.1 整数转罗马数字
- 6.2 罗马数字转整数
- 6.3 反转字符串
- 6.4 压缩字符串
- 6.5 验证回文串
- 6.6 长按键入
- 6.7 字符串中的第一个唯一字符
- 第七章 二分查找
- 7.1 猜数字大小
- 第八章 分治算法
- 第九章 动态规划
- 9.1 爬楼梯
- 9.2 使用最小花费爬楼梯
- 9.3 打家劫舍
- 9.4 打家劫舍 II
- 9.5 第N个泰波那契数
- 第十章 回溯算法
- 第十一章 栈
- 11.1 棒球比赛
- 第十二章 堆
- 12.1 数组中的第K个最大元素
- 第十三章 贪心算法
- 第十四章 排序
- 14.1 冒泡排序
- 14.2 鸡尾酒排序
- 14.3 选择排序
- 14.4 插入排序
- 14.5 折半插入排序
- 14.6 希尔排序
- 14.7 快速排序
- 14.8 树形选择排序
- 14.9 堆排序
- 第十五章 位运算
- 15.1 只出现一次的数字
- 第十六章 思维题
- 16.1 TinyURL 的加密与解密
- 16.2 灯泡开关
- 16.3 字母板上的路径
- 第十七章 队列
- 17.1 扑克牌顺序