java实现数组中的逆序对

 更新时间:2019年03月03日 16:41:47   作者:雨幕下的稻田  
这篇文章主要为大家详细介绍了java实现数组中的逆序对,具有一定的参考价值,感兴趣的小伙伴们可以参考一下

在数组中的两个数字,如果前面一个数字大于后面的数字,则这两个数字组成一个逆序对,例如在数组{7,5,6,4}中,一共存在5对逆序对,分别是{7,6},{7,5},{7,4},{6,4},{5,4}。输入一个数组,求出这个数组中的逆序对的总数P。并将P对1000000007取模的结果输出。,即输出P%1000000007。

代码

解法一

暴力简单低效,不会改变原数组

public static int inversePairs(int[] array) {
    if (array == null || array.length < 2) {
      return 0;
    }
    int count = 0;
    for (int i = 0; i < array.length; i++) {
      for (int j = i + 1; j < array.length; j++) {
        if (array[i] > array[j]) {
          count++;
        }
      }
    }
    return count % 1000000007;
  }


解法二

利用数组的归并排序,高效,但是会改变原数组

public static int inversePairs2(int[] array) {
    if (array == null || array.length < 2) {
      return 0;
    }
    int count = mergeSort(array, 0, array.length - 1);
    return count % 1000000007;
  }
 
  private static int mergeSort(int[] array, int start, int end) {
    if (start >= end) {
      return 0;
    }
 
    // 找到数组的中点,分割为两个子数组,递归求解
    int middle = (start + end) / 2;
    int left = mergeSort(array, start, middle);
    int right = mergeSort(array, middle + 1, end);
 
    // 存储归并后的数组
    int[] copy = new int[array.length];
    System.arraycopy(array, start, copy, start, end - start + 1);
    // 从两个子数组的尾部开始遍历
    int i = middle;
    int j = end;
    int copyIndex = end;
    // 记录逆序对的数量
    int count = 0;
 
    while (i >= start && j >= middle + 1) {
      // 数组是升序的
      // 如果左边数组比右边数组大,则将大的放入存储数组中
      // 并且累加逆序对,应为是有序的,所以左边数组的第i个元素比第j个及其之前的数都大
      if (array[i] > array[j]) {
        copy[copyIndex--] = array[i--];
        count += (j - middle);
      } else {
        copy[copyIndex--] = array[j--];
      }
    }
 
    // 将子数组剩余的部分一次写入归并后的存储数组
    while (i >= start) {
      copy[copyIndex--] = array[i--];
    }
    while (j >= middle + 1) {
      copy[copyIndex--] = array[j--];
    }
 
    // 将本次两个子数组的合并写入原数组中
    for (int k = start; k <= end ; k++) {
      array[k] = copy[k];
    }
    return left + right + count;
  }

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

相关文章

  • Java设计模式--代理模式

    Java设计模式--代理模式

    代理就是一个人或者一个机构代表另一个人或者另一个机构采取行动。在一些情况下,一个客户不想或者不能够直接引用一个对象,而代理对象可以在客户端和目标对象之前起到中介的作用
    2021-07-07
  • Spring Boot 通过注解实现数据校验的方法

    Spring Boot 通过注解实现数据校验的方法

    这篇文章主要介绍了Spring Boot 通过注解实现数据校验的方法,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-09-09
  • SpringBoot指定激活配置文件的方法

    SpringBoot指定激活配置文件的方法

    Spring Boot 对多环境整合已经有了很好的支持,能够在运行间、打包时自由切换环境,这篇文章主要介绍了SpringBoot指定激活配置文件,需要的朋友可以参考下
    2023-11-11
  • Java字符编码解码的实现详解

    Java字符编码解码的实现详解

    本篇文章介绍了,Java字符编码解码的实现详解。需要的朋友参考下
    2013-05-05
  • Java与Oracle实现事务(JDBC事务)实例详解

    Java与Oracle实现事务(JDBC事务)实例详解

    这篇文章主要介绍了Java与Oracle实现事务(JDBC事务)实例详解的相关资料,需要的朋友可以参考下
    2017-05-05
  • 详解SpringBoot中@ConditionalOnClass注解的使用

    详解SpringBoot中@ConditionalOnClass注解的使用

    这篇文章主要和大家详细介绍一下springboot中@ConditionalOnClass注解的用法,文中的示例代码讲解详细,感兴趣的小伙伴可以了解一下
    2022-08-08
  • Java中char[]输出不是内存地址的原因详解

    Java中char[]输出不是内存地址的原因详解

    这篇文章主要介绍了关于Java中char[]输出为什么不是内存地址的原因,文中通过示例代码介绍的很详细,需要的朋友们可以参考学习。
    2017-03-03
  • C++字符串的处理详解

    C++字符串的处理详解

    这篇文章主要介绍了C++ string字符串类,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-08-08
  • Java解决通信过程的中文乱码的问题

    Java解决通信过程的中文乱码的问题

    这篇文章主要介绍了 Java解决通信过程的中文乱码的问题的相关资料,需要的朋友可以参考下
    2017-01-01
  • SpringBoot连接Nacos集群报400问题及完美解决方法

    SpringBoot连接Nacos集群报400问题及完美解决方法

    这篇文章主要介绍了解决SpringBoot连接Nacos集群报400问题 ,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2022-02-02

最新评论