1164: 分治 逆序对

1164: 分治 逆序对

时间限制: 1 Sec   内存限制: 128 MB

题目描述

给一列数a1,a2,...,an,求它的逆序对数,即有多少个有序对(i,j),使得i<j且ai>aj。n可以高达10^6。

输入

第一行输入整数N(2<=N<=10^6).
接下来一行N个正整数数分别是a1,a2,...,an(ai<=10^6)。

输出

输出一个数表示逆序对数。

样例输入

4
2 4 3 1

样例输出

4

重点体会分治思想!!

要熟练会写,有一道叫做手套的题就考了逆序对,一会贴上。
这里直接粘代码了
[cpp]  view plain  copy
  1. #include<iostream>   
  2. #include<cstdio>   
  3. using namespace std;   
  4. int n,a[2000001],i,c[2000001];   
  5. long long ans;   
  6. void x(int l,int r)   
  7. {   
  8.     int mid=(l+r)/2,i,j,tmp;   
  9.     if(r>l)   
  10.     {   
  11.         x(l,mid);   
  12.         x(mid+1,r);   
  13.         tmp=l;   
  14.         for(i=l,j=mid+1;i<=mid&&j<=r;)   
  15.         {   
  16.             if(a[i]>a[j])   
  17.             {   
  18.                 c[tmp++]=a[j++];   
  19.                 ans+=mid-i+1;   
  20.             }   
  21.             else c[tmp++]=a[i++];   
  22.         }   
  23.         if(i<=mid) for(;i<=mid;) c[tmp++]=a[i++];   
  24.         if(j<=r) for(;j<=r;) c[tmp++]=a[j++];   
  25.         for(i=l;i<=r;i++) a[i]=c[i];   
  26.     }   
  27. }   
  28. int main()   
  29. {   
  30.     cin>>n;   
  31.     for(i=1;i<=n;i++) scanf("%d",&a[i]);   
  32.     x(1,n);   
  33.     cout<<ans;   
  34. }  
还有一份
[cpp]  view plain  copy
  1. #include<cstdio>    
  2. #include<iostream>    
  3. #include<algorithm>    
  4. using namespace std;    
  5. int A[9000010]={0},T[9000010]={0};   
  6. long long cnt=0;    
  7. void merge_sort(int x,int y)    
  8. {    
  9.     if(y-x>=1){    
  10.           int m=(y-x)/2+x;    
  11.           int p=x,q=m+1,i=x;    
  12.           merge_sort(x,m);    
  13.            merge_sort(m+1,y);    
  14.            while(p<=m||q<=y){    
  15.                 if(q>y||(p<=m&&A[p]<=A[q]))T[i++]=A[p++];    
  16.                 else {T[i++]=A[q++];cnt+=m+1-p;}    
  17.            }    
  18.            for(i=x;i<=y;i++)A[i]=T[i];    
  19.     }    
  20.           
  21. }    
  22. int main()    
  23. {    
  24.        int n,m,ans=0,i,k;    
  25.        scanf("%d",&n);    
  26.        for(i=0;i<n;i++)    
  27.        {    
  28.            scanf("%d",&A[i]);    
  29.                  
  30.        }    
  31.         merge_sort(0,n-1);    
  32.         printf("%lld",cnt);    
  33.              
  34. }   
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值