[洛谷P1908] 逆序对

题目描述 Description

猫猫TOM和小老鼠JERRY最近又较量上了,但是毕竟都是成年人,他们已经不喜欢再玩那种你追我赶的游戏,现在他们喜欢玩统计。最近,TOM老猫查阅到一个人类称之为“逆序对”的东西,这东西是这样定义的:对于给定的一段正整数序列,逆序对就是序列中ai>aj且i<j的有序对。知道这概念后,他们就比赛谁先算出给定的一段正整数序列中逆序对的数目。

输入输出格式 Input/output

输入格式:
第一行,一个数n,表示序列中有n个数。
第二行n个数,表示给定的序列。
输出格式:
给定序列中逆序对的数目。

输入输出样例 Sample input/output

样例测试点#1

输入样例:

6
5 4 2 6 3 1

输出样例:
11

说明 description

对于50%的数据,n≤2500
对于100%的数据,n≤40000。

分析:传统方法是树状数组,但是蒟蒻打起来比较困难,于是直接上归并排序。在合并的时候记录一下逆序对个数就好了。

#include<iostream>
#include<cstdio>
#include<cstring>
#include<cstdlib>
#include<algorithm>
#include<vector>
#include<cmath>
using namespace std;
int n,a[40001],h[40001],ans;
int read() //读入优化。
{
    char c=getchar();
    int a=0;
    while (c<‘0‘||c>‘9‘) c=getchar();
    while (c>=‘0‘&&c<=‘9‘)
    {
          a=a*10+c-‘0‘;
          c=getchar();
    }
    return a;
}
void merge_sort(int l,int r)
{
     if (l==r) return;
     int mid=(l+r)>>1;
     merge_sort(l,mid);
     merge_sort(mid+1,r);
     int left=l,right=mid+1;
     for (int i=l;i<=r;i++)
         if ((a[left]<a[right]||right>r)&&left<=mid)
         {
                                                    h[i]=a[left];
                                                    left++;
         }
         else
         {
             h[i]=a[right];
             right++;
             ans+=(mid-left+1); //记录逆序对个数
         }
     for (int i=l;i<=r;i++) a[i]=h[i];
}
int main()
{
    n=read();
    for (int i=1;i<=n;i++) a[i]=read();
    merge_sort(1,n);
    printf("%d",ans);
    return 0;
}    
时间: 2024-10-06 21:43:45

[洛谷P1908] 逆序对的相关文章

洛谷 P1908 逆序对 Label:归并排序||树状数组

题目描述 猫猫TOM和小老鼠JERRY最近又较量上了,但是毕竟都是成年人,他们已经不喜欢再玩那种你追我赶的游戏,现在他们喜欢玩统计.最近,TOM老猫查阅到一个人类称之为“逆序对”的东西,这东西是这样定义的:对于给定的一段正整数序列,逆序对就是序列中ai>aj且i<j的有序对.知道这概念后,他们就比赛谁先算出给定的一段正整数序列中逆序对的数目. 输入输出格式 输入格式: 第一行,一个数n,表示序列中有n个数. 第二行n个数,表示给定的序列. 输出格式: 给定序列中逆序对的数目. 输入输出样例 输

洛谷 P1908 逆序对

题目描述 猫猫TOM和小老鼠JERRY最近又较量上了,但是毕竟都是成年人,他们已经不喜欢再玩那种你追我赶的游戏,现在他们喜欢玩统计.最近,TOM老猫查阅到一个人类称之为“逆序对”的东西,这东西是这样定义的:对于给定的一段正整数序列,逆序对就是序列中ai>aj且i<j的有序对.知道这概念后,他们就比赛谁先算出给定的一段正整数序列中逆序对的数目. 输入输出格式 输入格式: 第一行,一个数n,表示序列中有n个数. 第二行n个数,表示给定的序列. 输出格式: 给定序列中逆序对的数目. 输入输出样例 输

洛谷 1908逆序对

P1908 逆序对 题目描述 猫猫TOM和小老鼠JERRY最近又较量上了,但是毕竟都是成年人,他们已经不喜欢再玩那种你追我赶的游戏,现在他们喜欢玩统计.最近,TOM老猫查阅到一个人类称之为"逆序对"的东西,这东西是这样定义的:对于给定的一段正整数序列,逆序对就是序列中ai>aj且i<j的有序对.知道这概念后,他们就比赛谁先算出给定的一段正整数序列中逆序对的数目. 输入输出格式 输入格式: 第一行,一个数n,表示序列中有n个数. 第二行n个数,表示给定的序列. 输出格式: 给

luogu P1908 逆序对

二次联通门 : luogu P1908 逆序对 /* luogu P1908 逆序对 权值线段树 + 离散化 + 指针版线段树... 把所有数离散化后将其作为下标建空树 对于每次插入的数字x, 查找x+1到max区间的数字存在的个数, 并将这个个数加入答案, 然后插入这个数字 最后输出答案 %%Menci的线段树...6到一定境界了... */ #include <algorithm> #include <cstdio> #define Max 40080 void read (i

洛谷P1908 求逆序对 [归并排序]

题目描述 猫猫TOM和小老鼠JERRY最近又较量上了,但是毕竟都是成年人,他们已经不喜欢再玩那种你追我赶的游 戏,现在他们喜欢玩统计.最近,TOM老猫查阅到一个人类称之为“逆序对”的东西,这东西是这样定义的:对于给定的一段正整数序列,逆序对就是序列中 ai>aj且i<j的有序对.知道这概念后,他们就比赛谁先算出给定的一段正整数序列中逆序对的数目. 输入输出格式 输入格式: 第一行,一个数n,表示序列中有n个数. 第二行n个数,表示给定的序列. 输出格式: 给定序列中逆序对的数目. 输入输出样例

AC日记——逆序对 洛谷 P1908

逆序对 思路: 线段树水过: 代码: #include <cstdio> #include <cstring> #include <iostream> #include <algorithm> using namespace std; #define maxn 40005 #define ll long long inline void in(int &now) { char Cget=getchar();now=0; while(Cget>'

P1908 逆序对-(cdq分治)

https://www.luogu.org/problem/P1908 沿用归并排序的思想求逆序对. 坑1:结果爆int型,需要用longlong 坑2:相对于归并排序,在比较的时候多了一个等号 举例说明归并排序解本题,例如有6个数, 36,87,99,   左区间范围是l到mid,下标用t1表示 1,2,50,     右区间范围是mid+1到r,下标用t2表示 分成2堆,两堆排好序,要合并.此时l=1,mid=3,t1=1; mid+1=4,r=6,t2=4; 比较36和1,选1,则左边还没

P1908 逆序对-(树状数组)

https://www.luogu.org/problem/P1908 比较喜欢线段树,懒得用树状数组(只会套模板,位运算的精髓没有领悟到),一直没有记录树状数组代码,又得捡回来,趁这道题记录一下模板,为三维偏序cdq套树状数组铺垫一下. 解题思路:先对原数组a从大到小排序,依次添加进树状数组c里,每次求前缀和的结果就是 当前数的逆序对的个数. 例如数据:55,44,22,66,33,11 初始化树状数组c,清0: 添加66到4号位,则添加数组为 0,0,0,1,0,0: 66前没有比它大的数,

luogu P1908 逆序对 |树状数组

题目描述 猫猫TOM和小老鼠JERRY最近又较量上了,但是毕竟都是成年人,他们已经不喜欢再玩那种你追我赶的游戏,现在他们喜欢玩统计.最近,TOM老猫查阅到一个人类称之为"逆序对"的东西,这东西是这样定义的:对于给定的一段正整数序列,逆序对就是序列中ai>aj且i<j的有序对.知道这概念后,他们就比赛谁先算出给定的一段正整数序列中逆序对的数目. Update:数据已加强. 输入格式 第一行,一个数n,表示序列中有n个数. 第二行n个数,表示给定的序列.序列中每个数字不超过10