求逆序数可以有三个方法,一个是模拟冒泡排序,效率低下;二是归并排序,求逆序数,一般够用;三是用树状数组,我还没研究过怎么用。
冒泡排序的算法:
for (int i = 0; i < t; i++) {
for (int j = 0; j < t - i - 1; j++)
if (a[j] > a[j + 1]) {
swap(a[j], a[j + 1]);
res++;//res就是要求的逆序数
}
}
合并排序的算法:
#include<iostream>
using namespace std;
#define MAX 1000000000
int a[500001];
int tmp[500001];
long long ans;//逆序数
void Mergesort(int b, int e)
{
if (e - b <= 0) return;
int mid = (b + e) / 2, p1 = b, p2 = mid + 1, i = b;
Mergesort(b, mid);
Mergesort(mid + 1, e);
while (p1 <= mid || p2 <= e)
{
if (p2 > e || (p1 <= mid && a[p1] <= a[p2]))
tmp[i++] = a[p1++];
else
{
tmp[i++] = a[p2++];
ans += (mid - p1 + 1);
}
}
for (i = b; i <= e; i++)a[i] = tmp[i];
}
int main()
{
int n;
while(cin>>n,n)
{
for(int i=1; i<=n; i++)
cin>>a[i];
ans=0;
Mergesort(1,n);
cout<<ans<<endl;
for(int i=1; i<=n; i++)cout<<a[i]<<" ";
cout<<endl;
}
return 0;
}
++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
刚刚把poj2299过掉,但是始终感觉1804过不掉不太爽。下到一个教案,说POJ上有个题,是必须用树状数组才能过的。等下准备去试一下。
在做2299的时候,最后的结果必须用长整形,发现在LINUX下,用NETBEANS,好像会编译不通过。不知道是什么原因。
由于这个类型用得很少,所以不太熟悉,还是在网上查了才知道怎么控制输入输出的。
要是用64位整形,直接这样用
__int64 res=0;
printf("%I64d/n",res);
要是LONG LONG形,可以这样用。
long long res;
printf("%u/n",res);
printf("%lld/n",res);