树状数组求逆序对

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了树状数组求逆序对相关的知识,希望对你有一定的参考价值。

我们知道,求逆序对最典型的方法就是归并排序,但是还有一种方法就是树状数组。假如你理解了树状数组,树状数组求逆序对相比归并排序排序要更好理解一些,而且树状数组的代码量也要少一些。

我们先看一下逆序对是什么吧。

  逆序对就是序列a中ai>aj且i<j的有序对。

 根据上面的定义我们很快的就可以写出O(n^2)的算法,即枚举j,再枚举所有小于j的i,统计ai>aj的数量。但这个算法当时间复杂度过高。

 如果我们能快速的统计出ai>aj的数量,时间复杂度就可以得到很好的提升。

树状数组就可以很好的做到这一点。

我们可以先开一个大小为a的最大值的数组t,每当读入一个数时,我们可以用桶排序的思想,将t[a[i]]加上1,然后我们统计t[1]~t[a[i]]的和ans,ans - 1(除掉这个数本身)就是在这个数前面有多少个数比它小。我们只要用i-ans就可以得出前面有多少数比它大,也就是逆序对的数量。

#include <bits/stdc++.h>
#define lowbit(x) (x)&(-x)
using namespace std;

const int maxn = 1e6+10;
int t[maxn],n,result;

void add(int x)
{
    while(x<=maxn)
    {
        t[x]++;
        x += lowbit(x);
    }
}

int query(int x)
{
    int ans=0;
    for (;x;x-=lowbit(x))
        ans+=t[x];
    return ans;
}

int main()
{
    int temp;
    scanf("%d",&n);
    for(int i=1;i<=n;i++)
    {
        scanf("%d",&temp);
        add(temp);
        result += i - query(temp);
    }
    printf("%d\n",result);
    return 0;
}

 

现在这个代码可以在数的最大值比较小的时候可以正确的得出答案,如果数据很大,这回造成我们要开的空间很大。

我们是否可以适当的减少空间的需求呢?我们看看下面这些数:

1 2 3 4 5 10

这6个数我们需要使用大小10的数组来存储,我们仔细想想,可以发现中间 6 7 8 9 这4个位置是没有用到的,也就是说这4个空间被浪费了。怎样减少这样的浪费呢?

我们可以在读完数数据后对他进行从小到大排序,我们用排完序的数组的下标来进行运算。这样可以保证小的数依旧小,大的数依旧大。这一步叫做离散化

struct Node
{
    int v;//数字本身
    int order;//原序列的的下标
}a[500005];

int reflect[500005];//用来存储原数第i个数的order下标是什么

//计算relect数组
relect[a[i].order] = i;

以上就是离散化的核心部分代码。

完整程序

代码来自:http://blog.csdn.net/SeasonJoe/article/details/50193789?locationNum=15&fps=1

#include <iostream>
#include <stdio.h>
#include <string.h>
#include <stdlib.h>
#include <algorithm>
using namespace std;

const int maxn= 500005;
int aa[maxn];//离散化后的数组
int c[maxn]; //树状数组
int n;

struct Node
{
    int v;
    int order;
}a[maxn];

bool cmp(Node a, Node b)
{
    return a.v < b.v;
}

int lowbit(int k)
{
    return k&(-k); //基本的lowbit函数 
}

void update(int t, int value)
{     //即一开始都为0,一个个往上加(+1),
    int i;
    for (i = t; i <= n; i += lowbit(i))
        c[i] += value;  
}

int getsum(int t)
{  //即就是求和函数,求前面和多少就是小于它的个数
    int i, sum = 0;
    for (i = t; i >= 1; i -= lowbit(i))
        sum += c[i];
    return sum;
}

int main()
{
    int i;
    while (scanf("%d", &n), n)
    {
        for (i = 1; i <= n; i++) //离散化
        {
            scanf("%d", &a[i].v);
            a[i].order = i;
        }
        sort(a + 1, a + n + 1,cmp);//从1到n排序,cmp容易忘
        memset(c, 0, sizeof(c));
        for (i = 1; i <= n; i++)
            aa[a[i].order] = i;
        __int64 ans = 0;
        for (i = 1; i <= n; i++)
        {
            update(aa[i], 1);
            ans += i - getsum(aa[i]); //减去小于的数即为大于的数即为逆序数
        }
        printf("%I64d\n", ans);
    }
    return 0;
}

 

以上是关于树状数组求逆序对的主要内容,如果未能解决你的问题,请参考以下文章

树状数组求逆序对

HDU 1394 Minimum Inversion Number (树状数组求逆序对)

树状数组求逆序对

求逆序对(树状数组)

Another Version of Inversion 二维树状数组求逆序对

树状数组求逆序对