51nod1019逆序数(归并排序/树状数组)
题目链接:http://www.51nod.com/onlineJudge/questionCode.html#!problemId=1019
题意:中文题诶~
思路:
方法1:归并排序~
归并排序过程为,先不断二分直至每组元素数目为一,此时我们可以将每组元素看做已排序状态;然后在回溯过程把这些组两两合并,并在合并过程中排序;
那么我们每一次合并都得到已排序的组,直至合并为一个组,即已排序数组;
那我们如何用上述过程统计逆序对数目呢~这就需要分析一下合并的具体过程啦;
假设我们现在有两个已排序数组a[p, m], a[q, y] 依次比较a[p], a[q],将其中较小的存入临时数组t中,那么最终得到的t数组就是a[p, y]区间所有元素的已排序状态啦,然后将t数组覆盖a[p, y]数组,那么a[p, y]就是已排序状态了啦~ 发现了有木有,如果我们在某一次操作中将右边的元素a[q]存入t中,那么a[q]大于当前左边数组a[p, m]的所有元素,即a[q]与数组a[p, m]中每个元素都能组成一个逆序对,对于a[q]元素我们可以得到(m-p)个逆序对 (数组a[p, m]区间为左闭右开即为区间
[p, m));对于数组a[p, m], a[q, y],假设元素gg, jj可以组成逆序对,我们可以将区间[p, y)里面可以组成逆序对的情况分为gg, jj都在 左边区间或者右边区间,gg, jj分别位于两个区间中,不过我们不难发现前两种情况就是合并前的第三种情况,例如对于数组a[p, m] 和a[q, y]我们求出gg, jj分处于两边的逆序对数为ans, 那么在下一次合并过程中,对于a[p', p] , a[p, q],则ans为gg, jj都位于a[p, q]中的逆序对数; 从这里我们可以发现虽然我们将逆序岁数分三种情况,事实上我们只要累计第三中情况的逆序对数就好啦~
代码:
#include <bits/stdc++.h>
#define MAXN 50010
using namespace std; int ans=; void teger_sort(int* a, int* t, int x, int y){
if(y-x>){ //**递归至单个元素为一组
int m=(y+x)>>;
int p=x, q=m, i=x;
teger_sort(a, t, x, m); //***左递归
teger_sort(a, t, m, y); //***右递归
while(p<m||q<y){ //**合并
if(q>=y||(p<m&&a[p]<=a[q])){
t[i++]=a[p++]; //**将左边元素复制到临时数组
}else{
t[i++]=a[q++]; //**将右边元素复制到临时数组
ans+=m-p; //**累计逆序对的数目
}
}
for(int i=x; i<y; i++){ //**将临时数组里已排序的元素还原到原数组
a[i]=t[i];
}
}
} int main(void){
int n, a[MAXN], t[MAXN];
scanf("%d", &n);
for(int i=; i<n; i++){
scanf("%d", &a[i]);
}
teger_sort(a, t, , n);
printf("%d\n", ans);
return ;
}
方法2:树状数组~
首先我们先建立一个数轴,对于输入的数据x,我们给数轴上的x标记1(初始时标记都为0啦~),数轴上x前面的数都比x小,也就是说x前面的所有标记的数都是不可以与x组成逆序对的数,假设x是第i个输入,用sum(x)表示x以及前面所有标记的和,即sum=x前面i-1个元素中比x小的个数+1(因为x本身不能和自己组成逆序对嘛),那么i-sum(x)就是x和其前面的元素可以组成的逆序对数咯,累加所有元素和其前面的元素组成的逆序对数就是我们要的答案啦~ 对于这里的数轴我们可以直接用数组标记就好了,不过这里的x数据范围是1e9,直接开数组肯定开不下啦,用map会超时,所以我们需要对输入的数据先hash一下~ 可是这个思路如果直接暴力的话还是会超时,不过,求sum(x)就是区间求和嘛,区间求和我们可以用树状数组或者线段树嘛,这里树状数组和线段树的效果一样,我们就用代码更简单的树状数组啦;要讲树状数组的话比较麻烦,这里就给出一个本人觉得讲的不错的树状数组博客好了~ http://blog.csdn.net/ljd4305/article/details/10101535 代码:#include <bits/stdc++.h>
#define MAXN 50010
using namespace std; int tree[MAXN], n; int lowbit(int x){ //**得到pow(2, k), 其中k为x的二进制末尾0的个数即x二进制最低位1的值
return x&(-x);
} void update(int x, int d){ //**向上更新父节点及根节点~
while(x<=n){
tree[x]+=d;
x+=lowbit(x);
}
} int sum(int x){ //**向下求和
int ans=;
while(x>){
ans+=tree[x];
x-=lowbit(x);
}
return ans;
} int main(void){
pair<int, int> p[MAXN];
int gg[MAXN], ans=;
scanf("%d", &n);
for(int i=; i<=n; i++){ //**hash
scanf("%d", &p[i].first);
p[i].second=i
}
sort(p+, p+n+);
for(int i=; i<=n; i++){
gg[p[i].second]=i;
}
for(int i=; i<=n; i++){
update(gg[i], ); //因为x不能和自己组成逆序对,要减去,所以我们要先更新再求和,更新后sum(x)就包括了x了嘛~
ans+=i-sum(gg[i]);
}
printf("%d\n", ans);
return ;
}
51nod1019逆序数(归并排序/树状数组)的更多相关文章
- 2018年全国多校算法寒假训练营练习比赛(第五场):A题:逆序数(树状数组or归并排序)
题目描述 在一个排列中,如果一对数的前后位置与大小顺序相反,即前面的数大于后面的数,那么它们就称为一个逆序.一个排列中逆序的总数就称为这个排列的逆序数.比如一个序列为4 5 1 3 2, 那么这个序列 ...
- 剑指 Offer 51. 数组中的逆序对 + 归并排序 + 树状数组
剑指 Offer 51. 数组中的逆序对 Offer_51 题目描述 方法一:暴力法(双层循环,超时) package com.walegarrett.offer; /** * @Author Wal ...
- CF #301 E:Infinite Inversions(逆序数,树状数组)
A-Combination Lock B-School Marks C-Ice Cave D-Bad Luck Island E-Infinite Inversions E:Infini ...
- H - 逆序数(树状数组)
在一个排列中,如果一对数的前后位置与大小顺序相反,即前面的数大于后面的数,那么它们就称为一个逆序.一个排列中逆序的总数就称为这个排列的逆序数. 如2 4 3 1中,2 1,4 3,4 1,3 1是逆序 ...
- NYOJ 117 求逆序数 (树状数组)
题目链接 描述 在一个排列中,如果一对数的前后位置与大小顺序相反,即前面的数大于后面的数,那么它们就称为一个逆序.一个排列中逆序的总数就称为这个排列的逆序数. 现在,给你一个N个元素的序列,请你判断出 ...
- AcWing 107. 超快速排序(归并排序 + 逆序对 or 树状数组)
在这个问题中,您必须分析特定的排序算法----超快速排序. 该算法通过交换两个相邻的序列元素来处理n个不同整数的序列,直到序列按升序排序. 对于输入序列9 1 0 5 4,超快速排序生成输出0 1 4 ...
- 洛谷 P1908 逆序对 Label:归并排序||树状数组 不懂
题目描述 猫猫TOM和小老鼠JERRY最近又较量上了,但是毕竟都是成年人,他们已经不喜欢再玩那种你追我赶的游戏,现在他们喜欢玩统计.最近,TOM老猫查阅到一个人类称之为“逆序对”的东西,这东西是这样定 ...
- POJ2299Ultra-QuickSort(归并排序 + 树状数组求逆序对)
树状数组求逆序对 转载http://www.cnblogs.com/shenshuyang/archive/2012/07/14/2591859.html 转载: 树状数组,具体的说是 离散化+树 ...
- Day2:T4求逆序对(树状数组+归并排序)
T4: 求逆序对 A[I]为前缀和 推导 (A[J]-A[I])/(J-I)>=M A[j]-A[I]>=M(J-I) A[J]-M*J>=A[I]-M*I 设B[]=A[]-M*( ...
随机推荐
- java之Timer
一.Java2的开发包中提供了一种很好使用的线程功能:你可以使用这些类创建后台进程,让其在等待一段规定的时间后执行,或者让其每隔一段时间执行.你也可以用Thread来完成,但利用Timer与Timer ...
- DAY3 python群发短信
手机轰炸,burpsuit 抓取注册页面输入的手机号,然后每点击一次forword ,都开开始放行,发短信.也可以发到repeat 里面进行 ,重复发送短信. import requests impo ...
- JavaScript判断图片是否加载完成的三种方式
JavaScript判断图片是否加载完成的三种方式 有时需要获取图片的尺寸,这需要在图片加载完成以后才可以.有三种方式实现,下面一一介绍. 一.load事件 1 2 3 4 5 6 7 8 9 10 ...
- Java 7 Concurrency Cookbook 翻译 第一章 线程管理之四
七.创建和运行一个后台线程 Java中有一种特别的线程叫做 deamon(后台) 线程.这类线程具有非常低的权限,并且只有在同一个程序中没有其他的正常线程在运行时才会运行.注意:当一个程序中只剩下后台 ...
- iOS开发——UI基础-提示框
提示框的种类有很多,废话不多说,直接上代码 一.文本提示框 运行结果如下: 代码实现如下: @interface ViewController () // 添加方法 - (IBAction)add; ...
- 爆料喽!!!开源日志库Logger的剖析分析
导读 Logger类提供了多种方法来处理日志活动.上一篇介绍了开源日志库Logger的使用,今天我主要来分析Logger实现的原理. 库的整体架构图 详细剖析 我们从使用的角度来对Logger库抽茧剥 ...
- BestCoder Round #86 解题报告
A.Price List Sol 求和查询 Code #include<cstdio> #include<algorithm> #include<iostream> ...
- net-snmp配置文件详解
net-snmp配置文件详解 net-snmp的配置文件是有一定的层次结构的,配置起来也很方便.网上找了很多资料,大概把这个配置文件的各个信息搞懂了一点.其实在net-snmp的EXAMPLE.con ...
- 4 Handler相关类——Live555源码阅读(一)基本组件类
这是Live555源码阅读的第一部分,包括了时间类,延时队列类,处理程序描述类,哈希表类这四个大类. Handler相关类概述 处理程序相关类一共有三个,其没有派生继承关系,但是其有友元关系和使用关系 ...
- django 实战 - eLeave Form
需求: 实现请假单的电子审批 1. 支持国际化 2. 支持模型级别的访问记录 here we go: 这里会写一系列的文章,来记录我实战的过程,由于接触django没多久,难免有疏漏之处,望拍砖不要太 ...