一、排序算法和时间复杂度
1 、最好情况、最坏情况、平均情况时间复杂度
2、时间复杂度的系数、常数 、低阶
3、比较次数和交换(或移动)次数
例子:比如我们有一组数据 2,9,3,4,8,3,按照大小排序之后就是 2,3,3,4,8,9
这组数据里有两个 3。经过某种排序算法排序之后,如果两个 3 的前后顺序没有改变,那我们就把这种排序算法叫作稳定的排序算法;如果前后顺序发生变化,那对应的排序算法就叫作不稳定的排序算法
- 稳定性意义:可针对对象的多种属性进行有优先级的排序
- 电商订单:按照金额大小对订单数据排序,对于相同金额的订单以下单时间早晚排序
原地排序:空间复杂度是 O(1) 的排序算法
二 、冒泡排序
冒泡排序只会操作相邻的两个数据,相邻两个元素进行比较。一次冒泡会让至少一个元素移动到它应该在的位置,重复 n 次,就完成了 n 个数据的排序工作
- 例子:对一组数据4,5,6,3,2,1进行从小到大排序
经过第一次冒泡排序,6这个元素已经存储在正确的位置上。进行6次这样的冒泡排序。
- 实现代码
// 冒泡排序,a表示数组,n表示数组大小
public void bubbleSort(int[] a, int n) {
if (n <= 1) return;
for (int i = 0; i < n; ++i) {
// 提前退出冒泡循环的标志位
boolean flag = false;
for (int j = 0; j < n - i - 1; ++j) {
if (a[j] > a[j+1]) { // 交换
int tmp = a[j];
a[j] = a[j+1];
a[j+1] = tmp;
flag = true; // 表示有数据交换
}
}
if (!flag) break; // 没有数据交换,提前退出
}
}
- 时间复杂度
- 有序度:数组中具有有序关系的元素对的个数
有序元素对:a[i] <= a[j], 如果i < j。
逆序元素对:a[i] > a[j], 如果i < j。
完全有序的数组的有序度叫作满有序度。
逆序度 = 满有序度 - 有序
三、插入排序
-
首先,我们将数组中的数据分为两个区间,已排序区间和未排序区间。初始已排序区间只有一个元素,就是数组的第一个元素。插入算法的核心思想是取未排序区间中的元素,在已排序区间中找到合适的插入位置将其插入,并保证已排序区间数据一直有序。重复这个过程,直到未排序区间中元素为空,算法结束
-
插入排序也包含两种操作,一种是元素的比较,一种是元素的移动。当我们需要将一个数据 a 插入到已排序区间时,需要拿 a 与已排序区间的元素依次比较大小,找到合适的插入位置。找到插入点之后,我们还需要将插入点之后的元素顺序往后移动一位,这样才能腾出位置给元素 a 插入
-
插入排序比冒泡排序更好?
冒泡排序中数据的交换操作:
if (a[j] > a[j+1]) { // 交换
int tmp = a[j];
a[j] = a[j+1];
a[j+1] = tmp;
flag = true;
}
插入排序中数据的移动操作:
if (a[j] > value) {
a[j+1] = a[j]; // 数据移动
} else {
break;
}