分治_归并_归并排序(逆序对)

912. 排序数组

上一次我们做这道题时用的是数组划分三块的思想 搭配随机选择基准元素的⽅法。

随机选择一个数,以这个数key为基准划分数组,小于key的数在左边,大于key的数在右边。再把被划分的两部份再找key值划分,直到只剩1或者0个元素返回。

遍历完最后一层该数组就排序完成
接下来我们采用归并的思想来做。

1.以中间点mid把数组分为两部份

2.一直划分直到 到最后一层只剩一个元素或者没有元素返回

3.**之下而上返回时,要对左右两部份数组进行排序。**借助一个临时数组 两个指针分别指向两个数组,由小到大排,先把小的放里面。再把临时数组排好的内容写回原数组。

4.以此类推,直到返回最上层,合并完左右两个有序数组,排序完成。

归并排序递归实现过程就像二叉树的后续遍历

class Solution {
    vector<int> tmp;
public:
    vector<int> sortArray(vector<int>& nums) {
        tmp.resize(nums.size());
        mergeSort(nums,0,nums.size()-1);
        return tmp;
    }
    void mergeSort(vector<int>&nums,int left,int right)
    {
        if(left>=right) return;
        //1.选择中间点划分区域
        int mid=(left+right)>>1;
        //2.把左右区间排序
        mergeSort(nums,left,mid);
        mergeSort(nums,mid+1,right);
        //3.合并两个有序数组
        int cur1=left,cur2=mid+1,i=0;
        while(cur1<=mid&&cur2<=right)
            tmp[i++]=nums[cur1]<nums[cur2]?nums[cur1++]:nums[cur2++];
        //合并剩余元素
        while(cur1<=mid) tmp[i++]=nums[cur1++];
        while(cur2<=right) tmp[i++]=nums[cur2++];
        //将tmp中排好序的数字写回nums
        for(int i=left;i<=right;i++)
            nums[i]=tmp[i-left];
    }
};

1.借助临时数组时,最好把它定义成全局变量,提前申请好空间,提高效率。

2.合并左右两数组时,可能有一方数组提前结束,后面还需要把另一个数组的剩余部分元素继续合并。

3.int mid=(left+right)>>1表示/2。如果数据溢出可以mid=left+(right-left)/2

LCR 170. 交易逆序对的总数

逆序对,第一个数>第二个数 且第一个数在数组的位置要在第二个数前面。

方法:归并

1.从一个数组中找逆序对,我们把该数组分为两半,左边部分和右半部分。

那逆序对的个数=左半部分逆序对的个数+右半部分逆序对的个数+左边取一个数和右边取一个数组成的逆序对的个数。

2.如果我们再计算完左右部分内逆序对的个数后,**对左右部分内进行排序,一左一右的逆序对个数会改变吗?**不会,因为左右两个数的相对位置没有改变,左半部分数仍在右半部分数的前面。一左一右挑完逆序对后,排序也不影响。

所以 总数=左半部分+排序 +右半部分+排序 +一左一右+排序

3.利用归并。算一个数组的逆序对个数,先把它分为左右两部份。求左部分的个数,是不是也可以把它再分为左右两部分,直到分到只剩1 0个元素 逆序对个数就为0。这和归并排序的规律是不是很像呢?

所以在递归过程中我们就可以把左右两部分的个数算出来。想一想,如果第一次原数组,分为左右部分,左部分的个数=它的左部分+它的右部分+它的一左一右。因此,我们只要算出一左一右的个数,第一层左部分 右部分的个数在递归中就可以算出来。

4.怎么算一左一右的个数?

首先为什么要排序呢?因为,当左右部分内排完序 左右部分都是升序/降序,进行归并时 当我们固定cur2 只要算出在左部分有多少个大于cur2的数就可以了。

策略一:找出该数前,有多少个数比我大

先来看升序时归并的代码,我们先规定cur2不动,找左部分大于cur2的个数。cur1++直到cur1>cur2,个数就为mid-cur1+1 (cur1右边),最后cur2++ 继续找。

1.nums[cur1]<=nums[cur2] cur1++;

2.nums[cur1]>nums[cur2] re+=mid-cur1+1; cur2++;

但我们反过来,先固定cur1不动,让cur2找比cur1小的个数。cur2++直到cur2>cur1,个数为cur2-(mid+1)+1 (cur2左边) ,最后cur1++ cur2++继续找。但cur2向右移,等再找到时,cur2左边的个数就包含的上一次找到的个数,就会重复。所以 升序时要先固定cur2

        while(cur1<=mid&&cur2<=right)
        {
            if(nums[cur1]<=nums[cur2]) tmp[i++]=nums[cur1++];
            else 
            {
                re+=mid-cur1+1;//cur1右边个数
                tmp[i++]=nums[cur2++];
            }
        }

策略二:当前元素的后面,有多少个元素比我小

降序也可以完成,不过我们先规定cur1不动,找右部分小于cur1的个数。cur2++直到cur2<cur1,个数就为right-cur2+1 (cur2右边),最后cur1++ 换下一个位置 继续找。

1.nums[cur1]<=nums[cur2] cur2++;

2.nums[cur1]>nums[cur2] re+=right-cur2+1; cur1++;

        while(cur1<=mid&&cur2<=right)
        {
            if(nums[cur1]<=nums[cur2]) tmp[i++]=nums[cur2++];
            else 
            {
                re+=right-cur2+1;//cur2右边的个数
                tmp[i++]=nums[cur1++];
            }
        }
class Solution {
    vector<int> tmp;
public:
    int reversePairs(vector<int>& record) {
        tmp.resize(record.size());
        return Sort(record,0,record.size()-1);
    }
    int Sort(vector<int>&nums,int left,int right)
    {
        int re=0;
        if(left>=right) return 0;
        //1.找中间数,把数组分为两半
        int mid=(left+right)>>1;
        //2.左边的个数+排序 右边的个数+排序
        re+=Sort(nums,left,mid);
        re+=Sort(nums,mid+1,right);
        //3.一左一右的个数 (升序)
        int cur1=left,cur2=mid+1,i=0;
        while(cur1<=mid&&cur2<=right)
        {
            if(nums[cur1]<=nums[cur2]) tmp[i++]=nums[cur1++];
            else 
            {
                re+=mid-cur1+1;
                tmp[i++]=nums[cur2++];
            }
        }
        while(cur1<=mid) tmp[i++]=nums[cur1++];
        while(cur2<=right) tmp[i++]=nums[cur2++];
        for(int i=left;i<=right;i++)
            nums[i]=tmp[i-left];
        return re;
    }
};

315. 计算右侧小于当前元素的个数

这道题也是算逆序对的个数,不过 上一道是算逆序对总数。这道是算在该下标下后面的数中可以组成逆序对的个数。

在排序中会把元素的下标改变,所以我们要有一个数组index记录元素在原数组的下标,并在排序过程中随着的元素同步改变 保证能找到该元素在原数组的下标。


我们以降序为基础,cur1固定 cur2++找cur2<cur1,right-cur2+1(cur2右边)的数就是可以和cur1组成逆序对的数,再根据index映射到cur1在原数组的下标。

class Solution {
    vector<int> tmp;
    vector<int> tmp2;
    vector<int> index;
    vector<int> re;
public:
    vector<int> countSmaller(vector<int>& nums) {
        int n=nums.size();
        re.resize(n);
        tmp.resize(n);
        tmp2.resize(n);
        index.resize(n);
        for(int i=0;i<n;i++) index[i]=i;
        sort(nums,0,n-1);
        return re;
    }
    void sort(vector<int>&nums,int left,int right)
    {
        if(left>=right) return;
        int mid=(right-left)/2+left;
        sort(nums,left,mid);
        sort(nums,mid+1,right);
        int cur1=left,cur2=mid+1,i=0;
        while(cur1<=mid&&cur2<=right)
        {
            if(nums[cur1]<=nums[cur2])
            {
                tmp2[i]=index[cur2];
                tmp[i++]=index[cur2++];
            }
            else 
            {
                re[index[cur1]]+=right-cur2+1;
                tmp2[i]=index[cur1];
                tmp[i++]=nums[cur1++];
            }
        }
        while(cur1<=mid)
        {
            tmp2[i]=index[cur1];
            tmp[i++]=nums[cur1++];
        }
        while(cur2<=right)
        {
            tmp2[i]=index[cur2];
            tmp[i++]=nums[cur2++];
        }
        for(int i=left;i<=right;i++)
        {
            nums[i]=tmp[i-left];
            index[i]=tmp2[i-left];
        }
    }
};

1.数组降序排序

2.index数组建立元素和它在原数组的下标的映射

3.index数组中的对应元素要在nums元素改变位置时同步改变

4.re[index[cur1]]+=right-cur2+1;

index[cur1]因为是固定cur1不动,进行映射时要找cur1

+= 在向上递归时 不断在合并的右数组中找逆序对

493. 翻转对

这道和找逆序对总个数的题不同的是,[n,m]前面n较大的值要>后面n小的值的2倍。

原本是n>m就可以了,这会导致无法在排序时一起找出翻转对。

1.我们要在排序前,利用单调性先找出翻转对的个数

2.再进行排序并合并 进行递归
依旧是把数组以中间点mid分为左右两部份,cur1 cur2分别指向左右部分。

1.降序 cur1固定 cur2++ 找cur2*2<cur1的点 re+=rgiht-cur2+1(cur2右边)

2.升序 cur2固定 cur1++ 找cur1/2>cur2的点 re+=mid-cur1+1(cur1右边)


降序 注意nums[cur2]*2时会出现整数溢出,可以选择用中间变量long long存取后再比较。

或者判断条件nums[cur1]<=nums[cur2]*2两边都除2.0 -> nums[cur1]/2.0<=nums[cur2]

为什么除2.0 而不是2?因为除整型2会除不尽

class Solution {
    vector<int> tmp;
public:
    int reversePairs(vector<int>& nums) {
        int n=nums.size();
        tmp.resize(n);
        return sort(nums,0,nums.size()-1);
    }
    int sort(vector<int>&nums,int left,int right)
    {
        if(left>=right) return 0;
        int re=0;
        int mid=(left+right)>>1;
        re+=sort(nums,left,mid);
        re+=sort(nums,mid+1,right);
        int cur1=left,cur2=mid+1,i=0;
        while(cur1<=mid&&cur2<=right)
        {
            long long c2=nums[cur2];
            c2*=2;
            if(nums[cur1]<=c2)
            {
                cur2++;
            }
            else
            {
                re+=right-cur2+1;
                cur1++;
            }
        }        
        cur1=left,cur2=mid+1,i=0;
        while(cur1<=mid&&cur2<=right)
        {
            if(nums[cur1]<=nums[cur2])
            {
                tmp[i++]=nums[cur2++];
            }
            else
            {
                tmp[i++]=nums[cur1++];
            }
        }
        while(cur1<=mid) tmp[i++]=nums[cur1++];
        while(cur2<=right) tmp[i++]=nums[cur2++];
        for(int i=left;i<=right;i++)
        {
            nums[i]=tmp[i-left];
        }
        return re;
    }
};

升序 改变cur1 cur2顺序就可以

class Solution {
    vector<int> tmp;
public:
    int reversePairs(vector<int>& nums) {
        int n=nums.size();
        tmp.resize(n);
        return sort(nums,0,nums.size()-1);
    }
    int sort(vector<int>&nums,int left,int right)
    {
        if(left>=right) return 0;
        int re=0;
        int mid=(left+right)>>1;
        re+=sort(nums,left,mid);
        re+=sort(nums,mid+1,right);
        int cur1=left,cur2=mid+1,i=0;
        while(cur1<=mid&&cur2<=right)
        {
            if(nums[cur1]/2.0<=nums[cur2])
            {
                cur1++;
            }
            else
            {
                re+=mid-cur1+1;
                cur2++;
            }
        }        
        cur1=left,cur2=mid+1,i=0;
        while(cur1<=mid&&cur2<=right)
        {
            if(nums[cur1]<=nums[cur2])
            {
                tmp[i++]=nums[cur1++];
            }
            else
            {
                tmp[i++]=nums[cur2++];
            }
        }
        while(cur1<=mid) tmp[i++]=nums[cur1++];
        while(cur2<=right) tmp[i++]=nums[cur2++];
        for(int i=left;i<=right;i++)
        {
            nums[i]=tmp[i-left];
        }
        return re;
    }
};
相关推荐
yuanManGan18 分钟前
数据结构漫游记:动态实现栈(stack)
数据结构
刘鑫磊up29 分钟前
第十二章:算法与程序设计
算法·算法与程序设计
羽落9631 分钟前
左神算法基础提升--3
算法
云边有个稻草人32 分钟前
【优选算法】三数之和(双指针算法)
笔记·算法·双指针算法
凌小添34 分钟前
Python入门教程丨2.3 流程控制、算法效率分析及优化
python·算法
像污秽一样41 分钟前
AI刷题-小R的随机播放顺序、不同整数的计数问题
开发语言·c++·算法
竹下为生1 小时前
LeetCode --- 432周赛
算法·leetcode·职场和发展
松桥爸(仁勇)1 小时前
【72课 局部变量与全局变量】课后练习
c++·算法
紫钺-高山仰止2 小时前
【脑机接口数据处理】matlab读取ns6 NS6 ns5NS5格式脑电数据
数据结构·数据库·算法·matlab
miilue2 小时前
[LeetCode] 链表完整版 — 虚拟头结点 | 基本操作 | 双指针法 | 递归
java·开发语言·数据结构·c++·算法·leetcode·链表