十大经典排序之:归并排序 |桶排序

Posted 菜菜bu菜

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了十大经典排序之:归并排序 |桶排序相关的知识,希望对你有一定的参考价值。

归并排序

归并排序原理

什么是归并排序呢?归并排序,首先要理解什么是归并:将两个的有序数列合并成一个有序数列,我们称之为"归并"。

思想:归并排序(Merge Sort)就是利用归并思想对数列进行排序。根据具体的实现,归并排序包括"从上往下"和"从下往上"2种方式。也是采用分治法的一个非常典型的应用。

算法实现

1、算法描述
Step 1:将n个元素分成两个含n/2元素的子序列(左边可能比右边多1个数)
Step 2:用MS将两个子序列递归排序,直到所有部分的元素个数都为1。(最后可以将整个原序列分解成n个子序列)
Step 3:从最底层开始逐步合并两个排好序的数列。

2、图示

3、算法空间复杂度和时间复杂度
时间复杂度

  • 最坏:o( n log ⁡ 2 n n\\log _2n nlog2n
  • 最好:o( n log ⁡ 2 n n\\log _2n nlog2n
  • 平均:o( n log ⁡ 2 n n\\log _2n nlog2n

空间复杂度(辅助存储):o(n)
稳定性:稳定

例题

用归并排序将以下数列按照从小到大的顺序输出:123,45,6,22,99,1,38,41,-6,0

java代码:


import java.util.Arrays;

public class Test 

    public static void mergeSort(int[] arr, int first, int last)
        if(null == arr || arr.length < 0 || first == last)
            return;
        
        mergeSort(arr, first, (last + first)/2);
        mergeSort(arr, (last + first)/2 + 1, last);
        sort(arr, first, (last + first)/2, last);

    

    //这个方法是合并两个有序数组,就好像直接插入排序算法一样
    public static void sort(int[] arr, int first, int mid, int last )
        int[] temp = new int[last - first + 1];
        int i = 0; 
        int j = 0; 
        int index = 0;
        while(i < mid - first + 1 && j < last - mid)
            if(arr[first + i] >= arr[mid + 1 + j])
                temp[index] = arr[mid + 1 + j];
                j++;
                index++;
             else 
                temp[index] = arr[first + i];
                i++;
                index++;
            
        
        while(i < mid - first + 1)
            temp[index] = arr[first + i];
            index++;
            i++;
        
        while(j < last - mid)
            temp[index] = arr[mid + 1 + j];
            index++;
            j++;
        
        for(int k = first, n = 0; k <= last; k++, n++)
            arr[k] = temp[n];
        

    

    public static void main(String[] args) 
        int[] arr=new int[]123,45,6,22,99,1,38,41,-6,0;
        //归并排序
        mergeSort(arr,0,9);

        System.out.println("归并排序后的结果是:");
        System.out.println(Arrays.toString(arr));
    



桶排序

桶排序原理

所谓的箱排序,是一个排序算法,工作的原理是将数组分到有限数量的桶子里。是一种又快又简便的方法,桶排序可以看做计数排序的升级版。

它将要排的数据分到多个有序的桶里,每个桶里的数据再单独排序,再把每个桶的数据依次取出,即可完成排序。如果使用稳定的内层排序,并且将元素插入桶中时不改变元素间的相对顺序,那么桶排序就是一种稳定的排序算法。

为了使桶排序更加高效,我们需要做到这两点:

  1. 在额外空间充足的情况下,尽量增大桶的数量
  2. 使用的映射函数能够将输入的 N 个数据均匀的分配到 K 个桶中

最快:当输入的数据可以均匀的分配到每一个桶中。
最慢:当输入的数据被分配到了同一个桶中。

算法实现

1、算法描述

  1. 设置一个定量的数组当作空桶;
  2. 遍历序列,并将元素一个个放到对应的桶中;
  3. 对每个不是空的桶进行排序;
  4. 从不是空的桶里把元素再放回原来的序列中。

2、图示

3、算法空间复杂度和时间复杂度
时间复杂度:

  • 最坏:o( n 2 n^2 n2
  • 最好:o( n n n)或者 o( n + k n+k n+k
  • 平均:o( n n n)或者 o( n + k n+k n+k

空间复杂度(辅助存储):o( n n n)或者 o( n + k n+k n+k

桶排序的平均时间复杂度为 o( n + n 2 / k + k n+n^2/k+k n+n2/k+k) (将值域平均分成 n 块 + 排序 + 重新合并元素),当k 约等于n 时为o( n n n

稳定性:稳定

例题

用桶排序将以下数列按照从小到大的顺序输出:
66,13,51,76,81,26,57,69,23

java代码:

import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collections;
import java.util.List;

public class Test 
    public static void bucketSort(int[] arr)
        int max = Integer.MIN_VALUE;
        int min = Integer.MAX_VALUE;
        for(int i = 0; i < arr.length; i++)
            max = Math.max(max, arr[i]);
            min = Math.min(min, arr[i]);
        

        //桶数
        int bucketNum = (max - min) / arr.length + 1;
        List<List<Integer>> bucketArr = new ArrayList<>(bucketNum);
        for(int i = 0; i < bucketNum; i++)
            bucketArr.add(new ArrayList<Integer>());
        

        //将每个元素放入桶
        for(int i = 0; i < arr.length; i++)
            int num = (arr[i] - min) / (arr.length);
            bucketArr.get(num).add(arr[i]);
        

        //对每个桶进行排序
        for(int i = 0; i < bucketArr.size(); i++)
            Collections.sort(bucketArr.get(i));
        

        //将各非空桶中的元素组合起来
        int k=0;
        for(int i=0;i<bucketNum;i++)
            if(bucketArr.get(i).size()!=0)
                for(int n:bucketArr.get(i))
                    arr[k]=n;
                    k++;
                
            
        
        

    

    public static void main(String[] args) 
        int[] arr=new int[]123,45,6,22,99,1,38,41,-6,0;
        //桶排序
        bucketSort(arr);

        System.out.println("桶排序后的结果是:");
        System.out.println(Arrays.toString(arr));
    



以上是关于十大经典排序之:归并排序 |桶排序的主要内容,如果未能解决你的问题,请参考以下文章

十大经典排序之:基数排序 |计数排序

[新星计划] Python手撕代码 | 十大经典排序算法

[新星计划] Python手撕代码 | 十大经典排序算法

十大经典排序(下)——Python3实现

十大经典排序算法

用 Python 实现十大经典排序算法