摘要java
选择排序的逻辑是先遍历比较出序列中最大的,而后把最大的放在最后位置。优化
遵循这个逻辑,用代码实现时,作到1.减小比较次数以外,这里引入一个新的指标 - 稳定性,2.保证排序过程当中的稳定性也是一个优化处理code
依据逻辑来看,最大值是放在尾部,并放置后,下次循环排除这个放置最大值的位置,for 循环从尾部开始最合适。排序
小循环开始前,须要先建立变量记录最大值坐标,这里使用的是 0 位置坐标,那么小循环开始时,就能够直接从 1 位置遍历,这就减小比较次数。class
for (int end = array.length-1; end > 0; end--) { int maxIndex = 0; for (int begin = 1; begin <= end; begin++) { if (cmp(maxIndex, begin) < 0) { maxIndex = begin; } } swap(maxIndex, end); }
开始前,先解释一下稳定性,稳定性是尽可能保持序列中两个元素在排序前和排序后的相对位置。好比下面伪代码:变量
// a1 与 a2 的值相等 a1 = a2 = 3 // 序列中 a1 值的位置在 a2 前面 array = [5, a1, 4, a2, 2] // 排序以后, a1 值位置在 a2 前面,保持了稳定性 array = [2, a1, a2, 4, 5]
序列中须要保证屡次排序后数据位置的相对稳定。好比信息表中,以 age 从小到大排序,不但愿 age 相等的一组数据中,它的名称在每一次排序以后都会有不一样的顺序。进阶
这里为了保证排序以后的稳定性,就当出现最大值时,也更新最大值的坐标。循环
为何这样就能够保证稳定性?遍历
首先最大值被交换到尾部以后,下次遍历比较的时候,就再也不比较这个位置,小循环的比较是从头开始的,若是出现等于最大值时,不更新最大值的位置,排序以后,相等的值,最靠前的值就被放在了最后面,改变了以前序列中相等值的相对位置。数据
for (int end = array.length-1; end > 0; end--) { int maxIndex = 0; for (int begin = 1; begin <= end; begin++) { if (cmp(maxIndex, begin) <= 0) { // 保证稳定性 maxIndex = begin; } } swap(maxIndex, end); }