java中归并排序和Master公式详解

作者:吃鱼的宗介 时间:2022-03-30 08:53:19 

基本思想

归并排序采取分治的思想进行排序,借用一张图片说明一下

java中归并排序和Master公式详解

将n个元素从中间切开,分成两部分。(左边可能比右边多1个数) 将步骤1分成的两部分,再分别进行递归分解。直到所有部分的元素个数都为1。 从最底层开始逐步合并两个排好序的数列。
优点在于,分治之后,合并排序的过程时间复杂度是O(N)(只需要扫描一遍就可以将两个有序的数组合并成一个有序数组)

实现

public static void MergeSort(int[] arr,int l , int r) {
       if (l == r || r < 0){
           return;
       }
       int middle = l+(r-l)/2; //取中值,可以防止达到Integer.MaxValue 溢出
       MergeSort(arr,l,middle);
       MergeSort(arr,middle+1,r);
       sort(arr,l,middle,r);
   }
   /**
    *
    * @param arr 等待排序的数组
    * @param l 左数组第一个指针
    * @param middle 分割左右数组
    * @param r 右数组最后一个指针
    */
   private static void sort(int[] arr, int l, int middle, int r) {
       int[] temp = new int[arr.length];
       System.arraycopy(arr, 0, temp, 0, arr.length);
       int right_first = middle+1;
       int tempIndex = l;
       while (l <= middle && right_first <= r){
           if (temp[tempIndex] < temp[right_first]){
               arr[l++] = temp[tempIndex++];
           }else {
               arr[l++] = temp[right_first++];
           }
       }
       while (tempIndex <= middle){
           arr[l++] = temp[tempIndex++];
       }
       while (right_first <= r ){
           arr[l++] = temp[right_first++];
       }

}

对数器验证

我们可以写个对数器,使用暴力排序的方式验证我们的排序方法是否准确

//生成1-100内随机数组
  public static int[] getParamArrays(){
       int[] result = new int[(int) (Math.random() * 100)];
       //随机生成数
       for (int i = 0; i < result.length; i++) {
           result[i] = (int) (Math.random() * 100);
       }
       return result;
   }
   public static void main(String[] args){
       for (int i = 0; i < 1000000; i++) {
           int[] nums = getParamArrays();
           int[] temp = nums;
           MergeSort(nums,0,nums.length-1);
           Arrays.sort(temp);
           //通过自定义比较次数,对随机数组进行排序验证正确性
           if (!nums.equals(temp)){
               System.out.println("wrong");
           }
       }
       System.out.println("end");
   }

递归时间复杂度计算 Master 公式

形如
T(N) = a * T(N/b) + O(N^d)(其中的a、b、d都是常数)
的递归函数,可以直接通过Master公式来确定时间复杂度
如果 log(b,a) < d,复杂度为O(N^d)
如果 log(b,a) > d,复杂度为O(N^log(b,a))
如果 log(b,a) == d,复杂度为O(N^d * logN)
此公式适用于子递归规模相等的情况下

a表示递归的次数也就是生成的子问题数,b表示每次递归是原来的1/b之一个规模,O(N^d) 表示分解和合并所要花费的时间之和(除开递归的复杂度)
此处就是 T(N)= 2*T(N/2)+O(N^1) 适用于第三种情况 复杂度为 O(nlogn)

来源:https://blog.csdn.net/qq_38895905/article/details/122337305

标签:java,归并排序,Master公式
0
投稿

猜你喜欢

  • 解决spring mvc 多数据源切换,不支持事务控制的问题

    2022-09-30 03:39:56
  • 基于C#实现的端口扫描器实例代码

    2022-04-29 20:21:56
  • Java解析使用JSON的多种方法

    2022-08-13 00:18:01
  • android canvas使用line画半圆

    2022-01-05 17:58:57
  • ResultSet如何动态获取列名和值

    2022-01-16 15:54:01
  • java小数位的例子

    2023-11-30 02:37:12
  • java时间戳转日期格式的实现代码

    2023-08-07 21:44:31
  • 如何使用两个栈实现队列Java

    2023-11-29 17:48:09
  • IDEA新建springboot项目时未生成pom.xml文件的解决操作

    2022-08-22 03:16:31
  • Android移动应用开发指南之六种布局详解

    2022-09-10 06:23:44
  • 详解Java中使用泛型实现快速排序算法的方法

    2022-04-28 09:47:00
  • 浅谈Spring中@NotEmpty、@NotBlank、@NotNull区别

    2023-01-02 08:15:49
  • C# XML基础入门小结(XML文件内容增删改查清)

    2022-10-18 17:12:02
  • Java获取文件的类型和扩展名的实现方法

    2021-09-24 08:56:56
  • C#实现char字符数组与字符串相互转换的方法

    2022-01-18 05:28:52
  • Android canvas drawBitmap方法详解及实例

    2021-06-12 12:17:18
  • Android实现仿Windows7图片预览窗格效果

    2022-10-14 01:48:04
  • Java 并发编程之线程挂起、恢复与终止

    2022-02-18 11:51:51
  • spring Boot查询数据分页显示的方法实例

    2023-04-28 13:30:08
  • Android混合开发教程之WebView的使用方法总结

    2023-03-11 21:27:03
  • asp之家 软件编程 m.aspxhome.com