hihocoder1524
描述
给定一个1-N的排列A1, A2, ... AN,如果Ai和Aj满足i < j且Ai > Aj,我们就称(Ai, Aj)是一个逆序对。
求A1, A2 ... AN中所有逆序对的数目。
输入
第一行包含一个整数N。
第二行包含N个两两不同整数A1, A2, ... AN。(1 <= Ai <= N)
对于60%的数据 1 <= N <= 1000
对于100%的数据 1 <= N <= 100000
输出
一个整数代表答案
- 样例输入
-
5
3 2 4 5 1 - 样例输出
-
5
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<iostream>
#include<vector>
#include<iomanip>
using namespace std; //1524 : 逆序对
const int maxn = 1e5 + ;
int n;
long long a[maxn], b[maxn];
int lowbit(int i)
{
return i&-i;
}
void update(int i)
{
while (i <= n)
{
b[i]++;
i += lowbit(i);
}
}
long long sum(int i)
{
long long cnt = ;
while (i)
{
cnt += b[i];
i -= lowbit(i);
}
return cnt;
}
int main()
{
cin >> n;
for (int i = ;i <= n;i++)
{
cin >> a[i];
}
memset(b, , sizeof(b));
long long res = ;
for (int i = n;i >= ;i--)
{
update(a[i]);
res += sum(a[i] - );
}
cout << res << endl;
return ;
}
树形数组这个东西还是比较有趣的。这个算法可以说是比较天才了。
具体的树形数组详解可以看这个博主的http://blog.csdn.net/ljd4305/article/details/10101535
归并排序的ac代码:
#include<cstdio>
#include<cstring>
#include<algorithm>
#include<iostream>
#include<vector>
#include<iomanip>
using namespace std; //1524 : 逆序对
const int maxn = 1e5 + ;
int n;
long long a[maxn],b[maxn];
long long cnt; void merge_sort(long long * a, int l, int r)
{
int mid = (l + r) >> ;
int m = l, n = mid + ;
int pos = l;
while (m<=mid&&n<=r)
{
if (a[m] < a[n])
{
b[pos++] = a[m++];
}
else
{
b[pos++] = a[n++];
cnt+=mid-m+;
}
} while (m <= mid) b[pos++] = a[m++];
while (n <= r) b[pos++] = a[n++];
for (int i = l;i <= r;i++)
a[i] = b[i];
}
void merge(long long * a, int l, int r)
{
if (l >= r) return;
int mid = (l + r) >>;
merge(a, l, mid);
merge(a, mid+, r);
merge_sort(a, l, r);
} int main()
{
cnt = ;
cin >> n;
for (int i = ;i <= n;i++)
{
cin >> a[i];
}
merge(a, , n);
cout << cnt << endl;
return ;
}
归并排序是将数列a[l,h]分成两半a[l,mid]和a[mid+1,h]分别进行归并排序,然后再将这两半合并起来。在合并的过程中(设l<=i<=mid,mid+1<=j<=h),当a[i]<=a[j]时,并不产生逆序数;当a[i]>a[j]时,在前半部分中比a[i]大的数都比a[j]大,将a[j]放在a[i]前面的话,逆序数要加上mid+1-i。因此,可以在归并排序中的合并过程中计算逆序数。
可以参考http://blog.csdn.net/acdreamers/article/details/16849761
hihocoder1524的更多相关文章
- 树形数组 java
2^k求法 int lowbit(int x) { return x&(-x); } lowbit()的返回值就是 2^k 次方的值. 基本树形数组的模板 import java.util.* ...
随机推荐
- 一条bash命令,清除指定的网络接口列表
在K8S的安装配置过程, 由于不断的测试, 会不断的生成各式各样的虚拟网络接口. 那么,不重新安装之前,清除前次产生的这些垃圾接口, 不让它们影响下次的测试,是很有必要的. 如何快速删除呢? 如下命令 ...
- SignalRMvc的简单例子
1.介绍 我们知道传统的http采用的是“拉模型”,也就是每次请求,每次断开这种短请求模式,这种场景下,client是老大,server就像一个小乌龟任人摆布,很显然,只有一方主动,这事情就没那么完美 ...
- Google搜索
https://www.google.com/intl/br/insidesearch/tipstricks/all.html 如何用好谷歌等搜索引擎?
- request.user哪里来的?
1.登录认证(auth认证登录后login后设置了session等信息包含用户的pk) >>>>> 2.用户再次请求登录的时候,通过 ...
- netty01(长短连接、java)
使用netty需要添加依赖包 netty版本:netty-5.0.0.Alpha2 http://files.cnblogs.com/files/applerosa/netty-5.0.0.Alpha ...
- Django中的Request和Response
接触Django这么久了,从来没有好好学习关于Django中的Request和Response对象.借着文件上传下载的相关工作,现在总结一下也不错. 当一个页面请求过来,Django会自动创建一个Re ...
- sublime text3 中 python3编译出现中文乱码解决方法
转 https://blog.csdn.net/Bin_bin_bingo/article/details/80531582 一.乱码现象 利用sublime自带编译快捷方式ctrl+B会出现中文乱码 ...
- Java数组实现循环队列的两种方法
用java实现循环队列的方法: 1.添加一个属性size用来记录眼下的元素个数. 目的是当head=rear的时候.通过size=0还是size=数组长度.来区分队列为空,或者队列已满. 2.数组中仅 ...
- Android XML shape 标签使用详解(apk瘦身,减少内存好帮手)
Android XML shape 标签使用详解 一个android开发者肯定懂得使用 xml 定义一个 Drawable,比如定义一个 rect 或者 circle 作为一个 View 的背景. ...
- BZOJ-2-4870: [Shoi2017]组合数问题 矩阵优化 DP
就 是 要 我 们 从 n k 件 物 品 里 面 选 出 若 干 件,使 得 其 数 量 模 k 等 于 r 的 方 案 数 . dp方程 f [ i , j ] 表示前 i 件物品拿了若干件使 ...