排序(冒泡、插入)

一、排序算法和时间复杂度

在这里插入图片描述

  • 分析排序算法执行效率

1 、最好情况、最坏情况、平均情况时间复杂度
2、时间复杂度的系数、常数 、低阶
3、比较次数和交换(或移动)次数

  • 排序算法的稳定性

例子:比如我们有一组数据 2,9,3,4,8,3,按照大小排序之后就是 2,3,3,4,8,9
这组数据里有两个 3。经过某种排序算法排序之后,如果两个 3 的前后顺序没有改变,那我们就把这种排序算法叫作稳定的排序算法;如果前后顺序发生变化,那对应的排序算法就叫作不稳定的排序算法

  1. 稳定性意义:可针对对象的多种属性进行有优先级的排序
  2. 电商订单:按照金额大小对订单数据排序,对于相同金额的订单以下单时间早晚排序
    在这里插入图片描述
  • 排序算法的内存消耗

原地排序:空间复杂度是 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;
}
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值