大厂敲门砖,常见的八大排序算法

目录

    • 一、算法思维导图
    • 二、算法分类
    • 三、冒泡排序
      • 1、基本思想
      • 2、动态效果图
      • 3、代码实现
      • 4、速度测试
    • 四、选择排序
      • 1、基本思想
      • 2、动态效果图
      • 3、代码实现
      • 4、速度测试
    • 五、插入排序
      • 1、基本思想
      • 2、动态效果图
      • 3、代码实现
      • 4、速度测试
    • 六、希尔排序
      • 1、基本思想
      • 2、效果图
      • 3、代码实例
    • 七、快速排序
      • 1、基本思想
      • 2、效果图
      • 3、算法描述
      • 4、代码实例
      • 5、速度测试
    • 八、归并排序
      • 1、基本思想
      • 2、效果图
      • 3、代码实现
      • 4、速度测试
    • 九、基数排序
      • 1、基本思想
      • 2、代码实例
      • 3、速度测试

一、算法思维导图

大厂敲门砖,常见的八大排序算法_第1张图片

二、算法分类

大厂敲门砖,常见的八大排序算法_第2张图片
算法的时间复杂度和空间复杂度:

大厂敲门砖,常见的八大排序算法_第3张图片

三、冒泡排序

1、基本思想

依次比较相邻的两个数,将比较小的数放在前面,比较大的数放在后面。

2、动态效果图

大厂敲门砖,常见的八大排序算法_第4张图片

3、代码实现

//冒泡排序
private static void bubbleSort(int[] arr){
    // 标识变量,表示是否进行过交换
    boolean flag = false;
    int temp = 0;
    for (int i = 0; i < arr.length-1; i++) {
        for (int j = 0; j < arr.length-1-i; j++) {
            // 如果前面的数比后面的数大,则交换
            if(arr[j]>arr[j+1]){
                flag = true;
                temp = arr[j];
                arr[j] = arr[j+1];
                arr[j+1] = temp;
            }
        }
        // 在一趟排序中,一次交换都没有发生过
        if(!flag){
            break;
        }
    }
}

4、速度测试

冒泡排序:120000数据,23秒

大厂敲门砖,常见的八大排序算法_第5张图片

四、选择排序

1、基本思想

(1)在序列中找到最小元素,放在第一个位置;

(2)从剩余未排序元素中继续寻找最小元素,放在第二个位置;

以此类推,直到排序完毕。

2、动态效果图

大厂敲门砖,常见的八大排序算法_第6张图片

3、代码实现

//选择排序
public static void selectSort(int[] arr) {
    for (int i = 0; i < arr.length - 1; i++) {
        int minIndex = i;
        int min = arr[minIndex];
        for(int j  = 1 + i;j<arr.length;j++){
            if(min > arr[j]){
                min = arr[j];
                minIndex = j;
            }
        }
        arr[minIndex] = arr[i];
        arr[i] = min;
    }
}

4、速度测试

选择排序:120000数据,4秒

大厂敲门砖,常见的八大排序算法_第7张图片

五、插入排序

1、基本思想

把n个待排序的元素第一位看成有序表,其它的看成无序表,排序过程中,每次从无序表中取出一个数,依次与有序表中的数进行比较,插入到合适的位置。

2、动态效果图

大厂敲门砖,常见的八大排序算法_第8张图片

3、代码实现

//插入排序
public static void insertSort(int[] arr ){
    int insertVal = 0;
    int insertIndex = 0;
    for (int i = 1; i < arr.length; i++) {
        //定义待插入的数
        insertVal = arr[i];
        // 即arr[i]的前面这个数的下标
        insertIndex = i - 1;
        // 给insertVal 找到插入的位置
        // 说明
        // 1. insertIndex >= 0 保证在给insertVal 找插入位置,不越界
        // 2. insertVal < arr[insertIndex] 待插入的数,还没有找到插入位置
        // 3. 就需要将 arr[insertIndex] 后移
        while(insertIndex >= 0 && insertVal < arr[insertIndex]){
            arr[insertIndex+1] = arr[insertIndex];
            insertIndex--;
        }
        // 当退出while循环时,说明插入的位置找到, insertIndex + 1
        if(insertIndex + 1 != i){
            arr[insertIndex+1] = insertVal;
        }
    }
}

4、速度测试

插入排序:120000数据,1秒

大厂敲门砖,常见的八大排序算法_第9张图片

六、希尔排序

1、基本思想

希尔排序也是一种插入排序,它是简单插入排序经过改进之后的一个更高效的版本,也称为缩小增量排序,同时该算法也是冲破O(n²)的第一批算法之一。它与插入排序的不同之处在于,它会优先比较较远的元素。

2、效果图

大厂敲门砖,常见的八大排序算法_第10张图片

3、代码实例

希尔排序有两种方式。

① 希尔排序(冒泡排序)

//希尔排序
// 使用逐步推导的方式来编写希尔排序
// 希尔排序时, 对有序序列在插入时采用交换法,
// 思路(算法) ===> 代码
public static void shellSort(int[] arr){
    // 根据前面的逐步分析,使用循环处理
    for(int step = arr.length/2;step>0;step /= 2 ){
        for (int i = step; i < arr.length; i++) {
            // 遍历各组中所有的元素(共step组,每组有个元素), 步长step
            for (int j = i - step; j >= 0; j -= step) {
                // 如果当前元素大于加上步长后的那个元素,说明交换
                if (arr[j] > arr[j + step]) {
                    int temp = arr[j];
                    arr[j] = arr[j + step];
                    arr[j + step] = temp;
                }
            }
        }
    }
}

速度测试:

冒泡法希尔排序:120000数据,11秒

大厂敲门砖,常见的八大排序算法_第11张图片
② 希尔排序(插入排序)

//对交换式的希尔排序进行优化->插入法
public static void shellSort2(int[] arr) {
// 增量step, 并逐步的缩小增量
for (int step = arr.length / 2; step > 0; step /= 2) {
    // 从第step个元素,逐个对其所在的组进行直接插入排序
    for (int i = step; i < arr.length; i++) {
        int j = i;
        int temp = arr[j];
        if(arr[j]<arr[j-step]){
            while (j - step >= 0&&temp<arr[j-step]){
                arr[j] = arr[j-step];
                j -= step;
            }
            arr[j] = temp;
        }
    }
}

速度测试:

插入法希尔排序:12000000数据,4秒,叹为观止

大厂敲门砖,常见的八大排序算法_第12张图片

七、快速排序

1、基本思想

通过一趟排序将待排记录分割成独立的两部分,其中一部分记录的关键字均比另一部分的关键字小,则可分别对这两部分记录进行排序,以达到整个排序的过程。

2、效果图

大厂敲门砖,常见的八大排序算法_第13张图片

3、算法描述

  • 快速排序使用分治法把一个串分为两个子串;
  • 找一个基准点,暂时选中间点为基准点;
  • 重新排序数列,比基准值小的放在基准点前面,大的放在后面;
  • 递归的把小于基准值的子数列和大于基准值的子数列排序;

4、代码实例

//快速排序
public static void quickSort(int[] arr,int left, int right) {
    int l = left; //左下标
    int r = right; //右下标
    //pivot 中轴值
    int pivot = arr[(left + right) / 2];
    int temp = 0; //临时变量,作为交换时使用
    //while循环的目的是让比pivot 值小放到左边
    //比pivot 值大放到右边
    while( l < r) {
        //在pivot的左边一直找,找到大于等于pivot值,才退出
        while( arr[l] < pivot) {
            l += 1;
        }
        //在pivot的右边一直找,找到小于等于pivot值,才退出
        while(arr[r] > pivot) {
            r -= 1;
        }
        //如果l >= r说明pivot 的左右两的值,已经按照左边全部是
        //小于等于pivot值,右边全部是大于等于pivot值
        if( l >= r) {
            break;
        }
 
        //交换
        temp = arr[l];
        arr[l] = arr[r];
        arr[r] = temp;
 
        //如果交换完后,发现这个arr[l] == pivot值 相等 r--, 前移
        if(arr[l] == pivot) {
            r -= 1;
        }
        //如果交换完后,发现这个arr[r] == pivot值 相等 l++, 后移
        if(arr[r] == pivot) {
            l += 1;
        }
    }
 
    // 如果 l == r, 必须l++, r--, 否则为出现栈溢出
    if (l == r) {
        l += 1;
        r -= 1;
    }
    //向左递归
    if(left < r) {
        quickSort(arr, left, r);
    }
    //向右递归
    if(right > l) {
        quickSort(arr, l, right);
    }
}

5、速度测试

快速排序:12000000数据,1秒,逆天而行
大厂敲门砖,常见的八大排序算法_第14张图片

八、归并排序

1、基本思想

归并排序采用经典的分治策略,分治法将问题分成一些小的问题然后递归解决,则治的阶段就是将分的阶段得到的答案修补在一起,即分而治之。

2、效果图

大厂敲门砖,常见的八大排序算法_第15张图片
大厂敲门砖,常见的八大排序算法_第16张图片

3、代码实现

//归并排序
public static void mergerSort(int[] arr,int left,int right,int[] temp){
    if(left<right){
        //中间索引
        int middle = (left + right)/2;
        //向左递归进行分解
        mergerSort(arr,left,middle,temp);
        //向右递归进行分解
        mergerSort(arr,middle + 1,right,temp);
        //合并
        merger(arr, left, middle, right, temp);
    }
}
 
//合并
public static void merger(int[] arr,int left,int middle,int right,int[] temp){
    int i = left; // 初始化i, 左边有序序列的初始索引
    int j = middle + 1; //初始化j, 右边有序序列的初始索引
    int t = 0; // 指向temp数组的当前索引
 
    //先把左右两边(有序)的数据按照规则填充到temp数组
    //直到左右两边的有序序列,有一边处理完毕为止
    while (i <= middle && j <= right) {//继续
        //如果左边的有序序列的当前元素,小于等于右边有序序列的当前元素
        //即将左边的当前元素,填充到 temp数组
        //然后 t++, i++
        if(arr[i] <= arr[j]){
            temp[t] = arr[i];
            t++;
            i++;
        }else { //反之,将右边有序序列的当前元素,填充到temp数组
            temp[t] = arr[j];
            t++;
            j++;
        }
    }
 
    //把有剩余数据的一边的数据依次全部填充到temp
    //左边的有序序列还有剩余的元素,就全部填充到temp
    while (i <= middle){
        temp[t] = arr[i];
        t++;
        i++;
    }
 
    //右边的有序序列还有剩余的元素,就全部填充到temp
    while (j <= right){
        temp[t] = arr[j];
        t++;
        j++;
    }
 
    //将temp数组的元素拷贝到arr
    //注意,并不是每次都拷贝所有
    t = 0;
    int tempLeft = left; //
    while (tempLeft <= right){
        arr[tempLeft] = temp[t];
        t++;
        tempLeft++;
    }
}

4、速度测试

归并排序:12000000数据,1秒,惊为天人
大厂敲门砖,常见的八大排序算法_第17张图片

九、基数排序

1、基本思想

将所有带比较数值统一为同样的数位长度,数据较短的数前面补0,然后从最低位开始依次进行依次排序,这样从最低位排序一直到最高位排序完成之后,数列就变成了一个有序序列。

2、代码实例

//基数排序
public static void radixSort(int arr[]){
    System.out.println("基数排序,arr长度:"+arr.length);
    //1. 得到数组中最大的数的位数
    int max = arr[0]; //假设第一数就是最大数
    for(int i = 1; i < arr.length; i++) {
        if (arr[i] > max) {
            max = arr[i];
        }
    }
    //得到最大数是几位数
    int maxLength = (max + "").length();
 
    //定义一个二维数组,表示10个桶, 每个桶就是一个一维数组
    //说明
    //1. 二维数组包含10个一维数组
    //2. 为了防止在放入数的时候,数据溢出,则每个一维数组(桶),大小定为arr.length
    //3. 名明确,基数排序是使用空间换时间的经典算法
    int[][] bucket = new int[10][arr.length];
 
    //为了记录每个桶中,实际存放了多少个数据,我们定义一个一维数组来记录各个桶的每次放入的数据个数
    //可以这里理解
    //比如:bucketElementCounts[0] , 记录的就是  bucket[0] 桶的放入数据个数
    int[] bucketElementCounts = new int[10];
 
    //这里我们使用循环将代码处理
    for(int i = 0 , n = 1; i < maxLength; i++, n *= 10) {
        //(针对每个元素的对应位进行排序处理), 第一次是个位,第二次是十位,第三次是百位..
        for(int j = 0; j < arr.length; j++) {
            //取出每个元素的对应位的值
            int digitOfElement = arr[j] / n % 10;
            //放入到对应的桶中
            bucket[digitOfElement][bucketElementCounts[digitOfElement]] = arr[j];
            bucketElementCounts[digitOfElement]++;
        }
        //按照这个桶的顺序(一维数组的下标依次取出数据,放入原来数组)
        int index = 0;
        //遍历每一桶,并将桶中是数据,放入到原数组
        for(int k = 0; k < bucketElementCounts.length; k++) {
            //如果桶中,有数据,我们才放入到原数组
            if(bucketElementCounts[k] != 0) {
                //循环该桶即第k个桶(即第k个一维数组), 放入
                for(int l = 0; l < bucketElementCounts[k]; l++) {
                    //取出元素放入到arr
                    arr[index++] = bucket[k][l];
                }
            }
            //第i+1轮处理后,需要将每个 bucketElementCounts[k] = 0 !!!!
            bucketElementCounts[k] = 0;
 
        }
    }
}

3、速度测试

基数排序:12000000数据,1秒,速度感人
大厂敲门砖,常见的八大排序算法_第18张图片

你可能感兴趣的:(数据结构与算法,排序算法,算法,数据结构)