求排列的逆序数(9018_1679)

为了做noip2013的火柴排序,特地练了一下逆序对的求法。

逆序对的求法,若要nlogn,有2种,一种就是用归并排序的思想。另一种,就是线段树或者树状数组。

这里,我采用了第一种。

#include<iostream>
#include<cstdio>
#include<algorithm>
#include<cstring>
using namespace std;
inline int read(){
    int t=1,num=0;char c=getchar();
    while(c>‘9‘||c<‘0‘){if(c==‘-‘)t=-1;c=getchar();}
    while(c>=‘0‘&&c<=‘9‘){num=num*10+c-‘0‘;c=getchar();}
    return num*t;
}
const int maxn=100010;
int a[maxn],b[maxn],n;long long ans=0;
void gb(int l,int mid,int r){
    int cnt=l,s=l,t=mid+1;
    while(s<=mid&&t<=r){
        if(a[s]<=a[t])b[cnt++]=a[s++];
        else{b[cnt++]=a[t++];ans+=mid-s+1;}
    }
    while(s<=mid)b[cnt++]=a[s++];
    while(t<=r)b[cnt++]=a[t++];
    for(int i=l;i<=r;i++)a[i]=b[i];
}
void gbsort(int l,int r){
    if(l==r)return;
    int mid=(l+r)>>1;
    gbsort(l,mid);gbsort(mid+1,r);gb(l,mid,r);
}
int main()
{
    n=read();
    for(int i=1;i<=n;i++)a[i]=read();
    gbsort(1,n);printf("%lld\n",ans);
    return 0;
}

本文由Yzyet编写,网址为www.cnblogs.com/Yzyet。非Yzyet同意,禁止转载,侵权者必究。

时间: 2024-10-24 03:55:25

求排列的逆序数(9018_1679)的相关文章

[树状数组]求排列的逆序数

求排列的逆序数 题目描述 在Internet上的搜索引擎经常需要对信息进行比较,比如可以通过某个人对一些事物的排名来估计他(或她)对各种不同信息的兴趣,从而实现个性化的服务. 对于不同的排名结果可以用逆序来评价它们之间的差异.考虑1,2,-,n的排列i1,i2,-,in,如果其中存在j,k,满足 j < k 且 ij > ik,那么就称(ij,ik)是这个排列的一个逆序. 一个排列含有逆序的个数称为这个排列的逆序数.例如排列 263451 含有8个逆序(2,1),(6,3),(6,4),(6,

《程序设计与算法(二)算法基础》《第五周 分治》求排列的逆序数 11

011:求排列的逆序数 查看 提交 统计 提问 总时间限制:  1000ms 内存限制:  65536kB 描述 在Internet上的搜索引擎经常需要对信息进行比较,比如可以通过某个人对一些事物的排名来估计他(或她)对各种不同信息的兴趣,从而实现个性化的服务. 对于不同的排名结果可以用逆序来评价它们之间的差异.考虑1,2,…,n的排列i1,i2,…,in,如果其中存在j,k,满足 j < k 且 ij > ik, 那么就称(ij,ik)是这个排列的一个逆序. 一个排列含有逆序的个数称为这个排

求排列的逆序数

考虑1,2,-,n (n <= 100000)的排列i1,i2,-,in,如果其中存在j,k,满足 j < k 且 ij > ik, 那么就称(ij,ik)是这个排列的一个逆序. 一个排列含有逆序的个数称为这个排列的逆序数.例如排列 263451 含有8个 逆序(2,1),(6,3),(6,4),(6,5),(6,1),(3,1),(4,1),(5,1),因此该排列的逆序数就是8. 现给定1,2,-,n的一个排列,求它的逆序数. 笨办法:O(n2) 分治O(nlogn): 1) 将数组分

求排列的逆序数之树状数组

代码实现 #include<iostream> #include<cstdio> #include<cstdlib> using namespace std; int num[100001]; int n,a[100001]; long long count=0; void add(int x){ for(int i=x;i<=n;i+=(i&-i))num[i]++; } void query(int p){ for(int i=p; i; i -= (

求字符数组逆序数(poj1007)

int InversionNumber(char* s,int len) { int ans=0;  //s逆序数 int A,C,G;  //各个字母出现次数,T是最大的,无需计算T出现次数 A=C=G=0; for(int i=len-1;i>=0;i--) { switch(s[i]) { case 'A':A++;break;  //A是最小的,无逆序数 case 'C': { C++; ans+=A;  //当前C后面出现A的次数就是这个C的逆序数 break; } case 'G':

[算法导论]练习2-4.d求排列中逆序对的数量

题目:给出一个确定在n个不同元素的任何排列中逆序对数量的算法,最坏情况需要Θ(nlgn)时间.(提示:修改归并排序.) 思路:修改从大到小排序的归并排序. 归并排序分为三步:分解.解决.合并. 分解:将排列A分解为A1.A2两个子排列. 解决:递归的从大到小排列A1和A2,在此同样递归的求解A1.A2的逆序对数量. 合并:按照递归排序的合并策略从大到小比较A1中的元素[a1,a2,a3…]和A2中的元素[b1,b2,b3…]. 1.若a1大于b1,则a1大于A2中的所有元素,逆序对的数量加上le

求逆序数(归并排序)

求逆序数 时间限制:2000 ms  |  内存限制:65535 KB 难度:5 描述 在一个排列中,如果一对数的前后位置与大小顺序相反,即前面的数大于后面的数,那么它们就称为一个逆序.一个排列中逆序的总数就称为这个排列的逆序数. 现在,给你一个N个元素的序列,请你判断出它的逆序数是多少. 比如 1 3 2 的逆序数就是1. 输入 第一行输入一个整数T表示测试数据的组数(1<=T<=5) 每组测试数据的每一行是一个整数N表示数列中共有N个元素(2〈=N〈=1000000) 随后的一行共有N个整

HDU1394 线段树求最小逆序数

题目:http://acm.hdu.edu.cn/showproblem.php?pid=1394 求最小的逆序数,在此贴下逆序数的概念: 在一个排列中,如果一对数的前后位置与大小顺序相反,即前面的数大于后面的数,那么它们就称为一个逆序.一个排列中逆序的总数就称为这个排列的逆序数.逆序数为偶数的排列称为偶排列:逆序数为奇数的排列称为奇排列.如2431中,21,43,41,31是逆序,逆序数是4,为偶排列. 也是就说,对于n个不同的元素,先规定各元素之间有一个标准次序(例如n个 不同的自然数,可规

逆序对问题---求逆序数

逆序数:在一个排列中,如果一对数的前后位置与大小顺序相反, 即前面的数大于后面的数,那么它们就称为一个逆序. 一个排列中逆序的总数就称为这个排列的逆序数.逆序数为偶数的排列称为偶排列:逆序数为奇数的排列称为奇排列. { 设 A 为一个有 n 个数字的有序集 (n>1),其中所有数字各不相同. 如果存在正整数 i, j 使得 1 ≤ i < j ≤ n 而且 A[i] > A[j],则 <A[i], A[j]> 这一个有序对称为 A 的一个逆序对,也称作逆序.逆序对的数量称作逆