【a703】求逆序对(线段树的解法)
Time Limit: 10 second
Memory Limit: 2 MB
问题描述
给定一个序列a1,a2...an。如果存在i小于j 并且ai大于aj,那么我们称之为逆序对,求给定序列中逆序对的数目
Input
第一行为n,表示序列长度,接下来的n行,第i+1行表示序列中的第i个数。
Output
所有逆序对的总数
Sample Input
4
3
2
3
2
Sample Output
3
【题解】
n的最大值为10W。
如果用线段树来解的话,求解的思路都相同。
我就直接复制前一篇的例子了。
比如
3 2 8 5
将它们排序(从大到小)
8 5 3 2
先把8放进去(原来的位置是3)
但是在放之前,先检查位置3之前有没有其他数字放进去了(如果放进去了肯定是比8大的数字,但是它们的下标又小于3(逆序对!));
因为没有
所以就把下标3对应的位置改为1;
即0 0 1 0
然后是第大的5(原来的位置是4)
则看看4前面有多少个元素已经放进去了(前缀和!)。
发现有1个。则答案递增1;
然后把arr[4] 改为1
即0 0 1 1
然后是第三大的数字3,它原来的位置是1,但是1前面没有数字已经放进去。则不递增答案。
最后是最小的元素2,它原来的位置就是2,然后位置2之前有一个数字3已经放在了位置1.即下标1的前缀和为1.则答案递增1.
最后答案为2;
这里实际上就是涉及到单节点的递增问题。
只要在改动的时候往上更新节点就可以了。很简单的线段树模型。
然后前缀和的话就是返回(1,pos)的和,线段树记录的是区间和的。
但是要注意一个问题。就是出现相同数字的情况。
则我们在写比较函数的时候,让相同的数字,之前的位置大的放在后面。然后我们处理到连续的相同数字的时候。就记录这是连续相同数字里面的第i个。
在累加完前缀和之后答案减去i-1,因为相同大小就不算逆序对了;
【代码】
#include <cstdio>
#include <algorithm> struct data2
{
int d, pos;
}; __int64 sum[400001] = { 0 },ans = 0; //线段树的数组貌似不用开4倍。。感觉2.2倍就够了。。
int n;
data2 a[100001];//以结构体的形式写,比较好写比较函数。 __int64 query(int l, int r, int begin, int end, int rt)//询问区间(l,r),当前节点的区间为(begin,end),当前节点为rt.
{
if (l > r)//如果询问的区间不合规则就退出
return 0;
if (l <= begin && end <= r) //如果这个节点所代表的区间在所要求的区间内,则返回这个区间的和。
{
return sum[rt];
}
int m = (begin + end) / 2;//获取这个节点所代表的区间的中点
__int64 re = 0;
if (l <= m)//如果左半部分有一部分区间在这个节点的左半部分则加上那一部分(当然会一直缩小区间直到找到那块区间为止)
re += query(l, r, begin, m, rt <<1);
if (m < r) //如果这个节点的区间的右半部分有一部分和所求区间有交集,则加上右半部分那段有交集的区间的和。
re += query(l, r, m+1, end, rt<<1|1);
return re;
} int cmp(const data2 &a, const data2 &b)//比较函数。
{
if (a.d > b.d)//第一关键是数字从大到小排序。
return 1;
if (a.d == b.d && a.pos < b.pos)//然后是如果数字相同则之前的位置大的在后面。
return 1;
return 0;
} void updata(int p, int num, int begin, int end, int rt)//要让数组下标为p的节点递增num,然后当前的节点所代表的区间为[begin,end],当前节点为rt
{//这里的p可以换成是区间理解为[p,p]
if (begin == end)//如果找到了这个节点p
{
sum[rt] += num;//在这个节点的和累加num
return;
}
int m = (begin + end) / 2;//取得这个节点所代表的区间中点
if (p <= m)//如果这个下标在中点的左边就往左递归节点
updata(p, num, begin, m, rt<<1);
else//否则往右递归节点
updata(p, num, m+1, end, rt<<1|1);
sum[rt] = sum[rt<<1] + sum[rt<<1|1];//因为子节点可能发生了改变,所以要更新当前节点的区间和。
} int main()
{
//freopen("F:\\rush.txt", "r", stdin);
//freopen("F:\\rush_out.txt", "w", stdout);
scanf("%d", &n);
for (int i = 1; i <= n; i++)
{
scanf("%d", &a[i].d);
a[i].pos = i; //记录原先的位置
}
std::sort(a + 1, a + 1 + n, cmp);//进行从大到小排序
int now = 0;
for (int i = 1; i <= n; i++)//假设当前是连续相同的数字中第x个,则now始终等于x-1
{
if (i != 1 && a[i - 1].d == a[i].d)
now++;
else
now = 0;
ans += query(1, a[i].pos - 1, 1, n, 1);
ans -= now;//因为会重复累加相同的数字,所以要减掉那几个相同的。
updata(a[i].pos, 1, 1, n, 1);//把a[i].pos递增1,然后要修改与之相关的区间的和。
}
printf("%I64d", ans);
return 0;
}
【a703】求逆序对(线段树的解法)的更多相关文章
- HDU 6318.Swaps and Inversions-求逆序对-线段树 or 归并排序 or 离散化+树状数组 (2018 Multi-University Training Contest 2 1010)
6318.Swaps and Inversions 这个题就是找逆序对,然后逆序对数*min(x,y)就可以了. 官方题解:注意到逆序对=交换相邻需要交换的次数,那么输出 逆序对个数 即可. 求逆序对 ...
- 逆序对 线段树&树状数组 (重制版)
逆序对的定义:长度为n的数组a,求满足i<j时a[i]>a[j]条件的数对个数. 第一次接触这种问题的人可能是更先想到的是n^2去暴力数前面有几个比他大的数. int main() { i ...
- 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*( ...
- 2019.01.22 bzoj3333: 排队计划(逆序对+线段树)
传送门 题意简述:给出一个序列,支持把ppp~nnn中所有小于等于apa_pap的'扯出来排序之后再放回去,要求动态维护全局逆序对. 思路:我们令fif_ifi表示第iii个位置之后比它大的数的个 ...
- hdu 4911 求逆序对数+树状数组
http://acm.hdu.edu.cn/showproblem.php?pid=4911 给定一个序列,有k次机会交换相邻两个位置的数,问说最后序列的逆序对数最少为多少. 实际上每交换一次能且只能 ...
- loj #535. 「LibreOJ Round #6」花火 树状数组求逆序对+主席树二维数点+整体二分
$ \color{#0066ff}{ 题目描述 }$ 「Hanabi, hanabi--」 一听说祭典上没有烟火,Karen 一脸沮丧. 「有的哦-- 虽然比不上大型烟花就是了.」 还好 Shinob ...
- 【bzoj3295】[Cqoi2011]动态逆序对 线段树套SBT
题目描述 对于序列A,它的逆序对数定义为满足i<j,且Ai>Aj的数对(i,j)的个数.给1到n的一个排列,按照某种顺序依次删除m个元素,你的任务是在每次删除一个元素之前统计整个序列的逆序 ...
- POJ 2299 Ultra-QuickSort 求逆序数 线段树或树状数组 离散化
我用的线段树写的. num数组表示已插入的数值的个数. 由于a[i]数值很大,但是n不是很大,所以要离散化处理 9 1 0 5 4 离散化后 4 1 0 3 2 这样保证最大值不会超过n #inclu ...
- 【a703】求逆序对(树状数组的解法)
Time Limit: 10 second Memory Limit: 2 MB 问题描述 给定一个序列a1,a2...an.如果存在i小于j 并且ai大于aj,那么我们称之为逆序对,求给定序列中逆序 ...
随机推荐
- Flume Channel Selectors官网剖析(博主推荐)
不多说,直接上干货! Flume Sources官网剖析(博主推荐) Flume Channels官网剖析(博主推荐) 一切来源于flume官网 http://flume.apache.org/Flu ...
- nuxt.js配置BASE_URL(基本域名)和NODE_ENV(环境变量)
一直以来,开发环境和生产环境的数据接口域名不一样总是困扰着我 每次打测试包或者线上包,我都得手动切换域名,我相信很多人的做法跟这差不多,类似下面这样: (你已经注意到,这个文件已经被我无情的删除了,因 ...
- sshfs 通过ssh 挂载远程目录
安装:yum -y install sshfs 挂载远程 ssh 文件系统: sshfs -p 1234 root@192.168.1.218:/home/ /mnt/ sshfs -p SSH端口 ...
- (转) 设置sqlplus中的退格键
转自:http://blog.itpub.net/26110315/viewspace-717249/ 有些时候当你使用sqlplus登录到数据库中的时候,敲错了命令想要删除修改的时候,发现以前敲入的 ...
- POJ 3134 - Power Calculus (IDDFS)
题意:求仅仅用乘法和除法最快多少步能够求到x^n 思路:迭代加深搜索 //Accepted 164K 1094MS C++ 840B include<cstdio> #include< ...
- 损失函数 - Andrew Ng机器学习公开课笔记1.2
线性回归中提到最小二乘损失函数及其相关知识.对于这一部分知识不清楚的同学能够參考上一篇文章<线性回归.梯度下降>. 本篇文章主要解说使用最小二乘法法构建损失函数和最小化损失函数的方法. 最 ...
- Win10系统如何设置所有程序默认以管理员身份运行?
原文:Win10系统如何设置所有程序默认以管理员身份运行? 在win10系统中有些用户发现一些程序只有使用管理员身份运行能才打开,这样的话就感觉会麻烦很多,那么有没有办法设置所有程序都默认以管理员身份 ...
- 【习题 5-8 UVA - 230】Borrowers
[链接] 我是链接,点我呀:) [题意] 在这里输入题意 [题解] 用map+set写个模拟就好. 3个区域 书架.桌子.别人的手上. 其中前两个区域的书都能借出去. [代码] #include &l ...
- 请求筛选模块被配置为拒绝包含 hiddenSegment 节的 URL 中的路径
转自原文 请求筛选模块被配置为拒绝包含 hiddenSegment 节的 URL 中的路径. 打开C:\Windows\System32\inetsrv\config路径 找到applicationH ...
- KDE Plasma 5.8 的 LTS 周期正好与其所采用的 Qt 5.6 的 LTS 周期一致
在 KDE Plasma 5.7 刚刚发布不久,KDE 开发团队就宣布了 KDE Plasma 5.8 的开发计划.这个版本将是一个 LTS 版本,据我所知,这应该是 KDE 历史上第一个 LTS 版 ...