• 反转对个数


    /**
     * 给定一个数组 nums ,如果 i < j 且 nums[i] > 2*nums[j] 我们就将 (i, j) 称作一个重要翻转对。
    
    你需要返回给定数组中的重要翻转对的数量。
    
     */
    
     /***
      * 没特别明白
        思路:1.原始数组分割子数组,分治求解
            solve(A)=solve(A1)+solve(A2)+cross(A1,A2),无限套娃直到子数组无法分割返回0
            2. 对于每个数组内,计算左边一半和右边一半比较的结果,因为1是基础,基础之后按照归并排序合并成2,所以后面的比较两边每一半都是有序的,所以如果nums[i]>2*nums[j],
            那么从i到mid的每个都大于j,减少计算
            3. 统计之后继续合并,下次对于4而言左右边的2都是有序的
    
      */
    public class reversepair {
        public static void main(String[] args) {
            int []nums={2,4,3,5,1};
            System.out.println(reversePairs(nums));
        }
        //暴力超时
        public static int reversePairs1(int[] nums) {
            int []count=new int[nums.length];
    
            for(int j=1;j<nums.length;j++){
                for(int i=0;i<j;i++){
                    if(nums[i]-nums[j]>nums[j]){
                        count[j]++;
                    }
                }
            }
            int num=0;
            for(int i=0;i<count.length;i++){
                num=num+count[i];
            }
            return num;
        }
        //归并排序
        public static int reversePairs(int[] nums) {
            if(nums.length<2){
                return 0;
            }
            return helper(nums,0,nums.length-1);
        }
        public static int helper(int[] nums,int left,int right) {
            if(left==right){
                return 0;
            }else{
                int mid=(left+right)/2;
                int n1=helper(nums,left,mid);
                int n2=helper(nums,mid+1,right);
                int ret=n1+n2;
                //统计下标对数量 这里数组是归并排序结束的,递增的
                int i=left;
                int j=mid+1;
                while(i<=mid){
                    while((j<=right)&&((long) nums[i] > 2 * (long) nums[j])){
                        j++;
                    }
                    ret=ret+j-mid-1;
                    i++;
                }
                //合并两个排序数组(对数组进行排序)
                int []sorted=new int[right-left+1];
                int p1=left;
                int p2=mid+1;
                int p=0;
                while(p1<=mid||p2<=right){
                    if(p1>mid){
                        sorted[p++]=nums[p2++];
                    }else if(p2>right){
                        sorted[p++]=nums[p1++];
                    }else{
                        if(nums[p1]<nums[p2]){
                            sorted[p++]=nums[p1++];
                        }else{
                            sorted[p++]=nums[p2++];
                        }
                    }
                }
                for(int k=0;k<sorted.length;k++){
                    nums[left+k]=sorted[k];
                }
                return ret;
            }
            
        }
    
        
    }
    

      

  • 相关阅读:
    软件错误,软件缺陷,软件故障与软件失效
    QEMU命令创建KVM Guest(bridge桥接)
    Linux下设置网卡随系统启动
    RHEL查看CPU等机器信息
    QEMU/KVM功能测试
    CentOS 删除自带的OpenJDK 和 安装SunJDK
    将Centos的yum源更换为国内的阿里云源
    Centos6.8下安装oracle_11gr2版主要过程
    Centos6.8下安装oracle_11gr2版主要过程
    opendrive.com提供的免费网盘
  • 原文地址:https://www.cnblogs.com/jieyi/p/14056093.html
Copyright © 2020-2023  润新知