lintcode:逆序对
题目
在数组中的两个数字如果前面一个数字大于后面的数字,则这两个数字组成一个逆序对。给你一个数组,求出这个数组中逆序对的总数。
概括:如果a[i] > a[j] 且 i < j, a[i] 和 a[j] 构成一个逆序对。
序列 [2, 4, 1, 3, 5] 中,有 3 个逆序对 (2, 1), (4, 1), (4, 3),则返回 3 。
解题
直接暴力找,时间复杂度O(n^2)
public class Solution {
/**
* @param A an array
* @return total of reverse pairs
*/
public long reversePairs(int[] A) {
// Write your code here
long res = 0;
int n = A.length;
for(int i=0;i<n;i++){
for(int j=i+1;j<n;j++){
if(A[i] >A[j]){
res ++;
}
}
}
return res;
}
}
归并排序的思想
对于数组A[p,...,q]
分成两个数组A[p,...,r],A[r+1,...,q]
当这两个数组都是有序的时候,其逆序对数很好求
可以两个指针i,j分布指向数组的尾部
当A[i]>A[j]时候一定是逆序对,注意是有序的,逆序对数量:j-(r+1) +1 = j-r
else 不是逆序对
程序
public class Solution {
/**
* @param A an array
* @return total of reverse pairs
*/
long res = 0;
public long reversePairs(int[] A) {
// Write your code here
int n = A.length;
reversP(A,0,n-1);
return res;
}
public void reversP(int[] A,int low,int high){
if(low>=high)
return;
int mid = low + (high - low)/2;
reversP(A,low,mid);
reversP(A,mid+1,high);
merge(A,low,mid,high);
}
public void merge(int[] A,int low ,int mid ,int high){
int len = high - low + 1;
int[] C = new int[len]; // 临时存放中间归并数组
int i = mid;
int j = high;
int k = len -1;
while(i>= low && j>=mid+1){
if(A[i] > A[j]){
C[k--] = A[i];
i--;
res += j - (mid + 1) +1; // 逆序对数量
}else{
C[k--] = A[j];
j--;
}
}
while(i>=low){
C[k--] = A[i];
i--;
}
while(j>=mid+1){
C[k--] = A[j];
j--;
}
for(k=0;k<len;k++){
A[k+low] = C[k];
}
}
public void print(int[] A){
for(int a:A){
System.out.print(a+"\t");
}
System.out.println();
}
}
分析下输出过程
测试样例:[2,4,1,3,5]
5个元素划分的区间
元素下标上界是4
[0,4]划分[0,2]、[3,4]
---[0,2]划分:[0,1]、[2]
-------[0,1]划分:[0]、[1]
---[3,4]划分:[3]、[4]
输出情况
2 4 1 3 5 最底层只有一个元素
1 2 4 3 5 [0,2]合并
1 2 4 3 5
1 2 3 4 5 [0,4]合并
算法可行性
一个数组分成B、C两部分,B、C两部分分布升序排序
对于A[i] 在B中,A[j]在C中的情况,关于A[i]与A[j]的逆序对数量与B中A[i]的位置、C中A[j]的位置无关,这个很显然
程序开始的时候找到是1个元素,后来合并成两个元素的数组,这样慢慢的合并,并计算逆序对的数量,最后就得到答案了
lintcode:逆序对的更多相关文章
- 【CQOI2011】动态逆序对 BZOJ3295
Description 对于序列A,它的逆序对数定义为满足i<j,且Ai>Aj的数对(i,j)的个数.给1到n的一个排列,按照某种顺序依次删除m个元素,你的任务是在每次删除一个元素之前统计 ...
- CH Round #72 奇数码问题[逆序对 观察]
描述 你一定玩过八数码游戏,它实际上是在一个3*3的网格中进行的,1个空格和1~8这8个数字恰好不重不漏地分布在这3*3的网格中. 例如:5 2 81 3 _4 6 7 在游戏过程中,可以把空格与其上 ...
- POJ3928Ping pong[树状数组 仿逆序对]
Ping pong Time Limit: 1000MS Memory Limit: 65536K Total Submissions: 3109 Accepted: 1148 Descrip ...
- NOIP2013火柴排队[逆序对]
题目描述 涵涵有两盒火柴,每盒装有 n 根火柴,每根火柴都有一个高度. 现在将每盒中的火柴各自排成一列, 同一列火柴的高度互不相同, 两列火柴之间的距离定义为: ∑(ai-bi)^2 其中 ai 表示 ...
- bzoj 3295 动态逆序对 CDQ分支
容易看出ans[i]=ans[i-1]-q[i],q[i]为删去第i个数减少的逆序对. 先用树状数组算出最开始的逆序对,预处理出每个数前边比它大的和后边比它小的,就求出了q[i]的初始值. 设b[i] ...
- 诸城模拟赛 dvd的逆序对
[题目描述] dvd是一个爱序列的孩子. 他对序列的热爱以至于他每天都在和序列度过 但是有一个问题他却一直没能解决 给你n,k求1~n有多少排列有恰好k个逆序对 [输入格式] 一行两个整数n,k [输 ...
- 归并求逆序数(逆序对数) && 线段树求逆序数
Brainman Time Limit: 1000 MS Memory Limit: 30000 KB 64-bit integer IO format: %I64d , %I64u Java c ...
- BZOJ 3295: [Cqoi2011]动态逆序对
3295: [Cqoi2011]动态逆序对 Time Limit: 10 Sec Memory Limit: 128 MBSubmit: 3865 Solved: 1298[Submit][Sta ...
- BZOJ 3295 【Cqoi2011】 动态逆序对
Description 对于序列\(A\),它的逆序对数定义为满足\(i<j\),且\(A_i>A_j\)的数对\((i,j)\)的个数.给\(1\)到\(n\)的一个排列,按照某种顺序依 ...
随机推荐
- HTML5七大优势“逼宫”APP
HTML5颠覆了PC互联网的格局,优化了移动互联网的体验,接下来几年,HTML5将颠覆原生App世界. 跨平台: 在多屏年代,开发者的痛苦指数非常高,人人都期盼HTML5能扮演救星.多套代码.不同技术 ...
- Swift TabeleViewCell dequeueReusableCellWithIdentifier 使用的新的细节,原来现在可以这样
今天在看官方的TableView Guide,突然想起来最近写的一个代码中实现tableViewCell复用的时候有点问题: var cell = UITableViewCell(style: UIT ...
- [转]Linux Ubuntu上架设FTP
Linux Ubuntu上架设FTP http://www.blogjava.net/stonestyle/articles/369104.html 操作系统:ubuntu (GNU/Linux) 为 ...
- Labview实现单边带信号调制(SSB)[移相法]
Labview实现单边带信号调制(SSB)[移相法] 时域上的表达式为 调制器模型为 这个实验中需要相位偏移比较多,因为一共用了四个信号仿真器,一个是无偏移的调制信号,一个是偏移的调制信号,一个是无偏 ...
- java 静态变量生命周期(类生命周期)
Static: 加载:java虚拟机在加载类的过程中为静态变量分配内存. 类变量:static变量在内存中只有一个,存放在方法区,属于类变量,被所有实例所共享 销毁:类被卸载时,静态变量被销毁,并释放 ...
- Mysql的相关命令
1.登录服务器 mysql -h host -u user -p mysql -h host -u user -p 数据库 2.使用SHOW语句找出服务器上当前存在什么数据库:mysql> SH ...
- VRP-Lua学习笔记
至于vrp是什么东西以及为什么要学习vrp,vrp的简单操作这些问题请自行右转抵拢倒拐找百度或者去中视典官网去找教程,我这里不会在赘述. 今天默认我们已经会使用VRP的脚本编辑器,用其他语言来为VRP ...
- 新安装Ubuntu加载时提示“为/检查磁盘时发生严重错误”的解决方法
本文部分内容转载自: http://jingyan.baidu.com/article/0aa22375bbffbe88cc0d6419.html http://www.aichengxu.com/v ...
- java并发编程:进程和线程
java并发编程涉及到很多内容,当然也包括多线程,再次补充点相关概念 原文地址:http://www.cnblogs.com/dolphin0520/p/3910667.html 一.操作系统中为什么 ...
- selenium--嵌套frame定位
网页源码: 案例1 :iframe有id.name属性 网页上有3个frame:header.menu.main,分别代码顶部.左侧.右侧(其中menu.main在另外一个frameset中) 如何定 ...