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;

然后把tree_arr[4] 改为1

即0 0 1 1

然后是第三大的数字3,它原来的位置是1,但是1前面没有数字已经放进去。则不递增答案。

最后是最小的元素2,它原来的位置就是2,然后位置2之前有一个数字3已经放在了位置1.即下标1的前缀和为1.则答案递增1.

最后答案为2;

而树状数组就是专门处理前缀和的。

排序完毕之后,从大到小,从他们各自原来的位置,一直往左累加长条(前缀和)就可以了。然后累加完毕之后,需要把它放在原来的位置(置为1),则又要往右更新了。

但是要注意一个问题。就是出现相同数字的情况。

则我们在写比较函数的时候,让相同的数字,之前的位置大的放在后面。然后我们处理到连续的相同数字的时候。就记录这是连续相同数字里面的第i个。

在累加完前缀和之后答案减去i-1,因为相同大小就不算逆序对了;

【代码】

#include <cstdio>
#include <algorithm> struct data2
{
int d, pos;
}; int n;
__int64 tree_arr[100001] = { 0 };
__int64 ans = 0;
data2 a[100001]; int lowbit(int x) //求整数x的二进制的最靠近右边的数字1所代表的数字。
{
return x & -x; //-x是在x的二进制按位取反之后再加上1的结果。然后和x取与运算。
} 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 find(int now) //一直往左累加长条(区间和)
{
if (now <= 0)//越界了则退出
return;
ans += tree_arr[now];//累加这一段的区间和
find(now - lowbit(now));//一直往左
} void add(int now)//把数组中now位置上的数字递增1
{
if (now > n)
return;
tree_arr[now] ++;
add(now + lowbit(now));//然后继续往左,去递增那些会受到影响的区间。
} 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); //输入n个数据,并且记录它们原来的位置。
a[i].pos = i;
}
std::sort(a + 1, a + 1 + n, cmp);//把a数组以数字大小为关键字从大到小排序。
int now = 0;
for (int i = 1; i <= n; i++)//把每一个数字放到树状数组中。
{
if (i != 1 && a[i - 1].d == a[i].d)//如果出现了连续相同的数字就记录这是连续相同数字里的第x个,now=x-1;
now++;
else
now = 0;
find(a[i].pos);//往左累加前缀和
ans -= now;//减去重复的
add(a[i].pos);//修改与其相关的区间。
}
printf("%I64d", ans);
return 0;
}

【a703】求逆序对(树状数组的解法)的更多相关文章

  1. 求逆序对[树状数组] jdoj

    求逆序对 题目大意:给你一个序列,求逆序对个数. 注释:n<=$10^5$. 此题显然可以跑暴力.想枚举1到n,再求在i的后缀中有多少比i小的,统计答案即可.这显然是$n^2$的.这...显然过 ...

  2. [BZOJ 3295] [luogu 3157] [CQOI2011]动态逆序对(树状数组套权值线段树)

    [BZOJ 3295] [luogu 3157] [CQOI2011] 动态逆序对 (树状数组套权值线段树) 题面 给出一个长度为n的排列,每次操作删除一个数,求每次操作前排列逆序对的个数 分析 每次 ...

  3. Bzoj 2141: 排队 分块,逆序对,树状数组

    2141: 排队 Time Limit: 4 Sec  Memory Limit: 259 MBSubmit: 1310  Solved: 517[Submit][Status][Discuss] D ...

  4. luogu1908 逆序对 树状数组

    题目大意:对于给定的一段正整数序列,逆序对就是序列中ai>aj且i<j的有序对.求一段序列的逆序对数. 对于一个数组T,其一个点的值为值与该点下标相等的A序列中点的个数.对T维护一个树状数 ...

  5. P1908 逆序对——树状数组&离散化&快读快写の学习

    题目简述: 对于给定的一段正整数序列,逆序对就是序列中 a_i>a_jai​>aj​ 且 i<ji<j 的有序对. 输出序列中逆序对的数目. 知识补充: 树状数组: 这东西就是 ...

  6. 洛谷 P1908 逆序对(树状数组解法)

    归并排序解法:https://www.cnblogs.com/lipeiyi520/p/10356882.html 题目描述 猫猫TOM和小老鼠JERRY最近又较量上了,但是毕竟都是成年人,他们已经不 ...

  7. BZOJ - 3295 动态逆序对 (树状数组套treap)

    题目链接 思路和bzoj2141差不多,不过这道题的数据更强一些,线段树套treapT了,树状数组套treap卡过~~ #include<bits/stdc++.h> using name ...

  8. luogu P1908 逆序对 |树状数组

    题目描述 猫猫TOM和小老鼠JERRY最近又较量上了,但是毕竟都是成年人,他们已经不喜欢再玩那种你追我赶的游戏,现在他们喜欢玩统计.最近,TOM老猫查阅到一个人类称之为"逆序对"的 ...

  9. ACM学习历程—HDU5592 ZYB's Premutation(逆序数 && 树状数组 && 二分)(BestCoder Round #65 1003)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=5592 题目大意就是给了每个[1, i]区间逆序对的个数,要求复原原序列. 比赛的时候2B了一发. 首先 ...

随机推荐

  1. 洛谷——P1101 单词方阵

    https://www.luogu.org/problem/show?pid=1101#sub 题目描述 给一nXn的字母方阵,内可能蕴含多个“yizhong”单词.单词在方阵中是沿着同一方向连续摆放 ...

  2. Spring入门--控制反转(IOC)与依赖注入(DI)

        1.控制反转(Inversion of Control)与依赖注入(Dependency Injection) 控制反转即IoC (Inversion of Control).它把传统上由程序 ...

  3. C# mongodb帮助类

    这是在C#连接MongoDB的帮助类,所使用的驱动是在Vs2015的Nuget管理器中下载的mongodb驱动. 下载第一个,会自动下载下面的两个,不要删除. 在配置文件中配置连接字符串connStr ...

  4. NYOJ 552 小数阶乘

    小数阶乘 时间限制:1000 ms  |  内存限制:65535 KB 难度:1 描写叙述 编写一个程序,求一个数m的阶乘. 输入 有多组測试数据,以EOF结束. 每组測试数据有1个整数m. 输出 每 ...

  5. php课程 11-37 类和对象的关系是什么

    php课程 11-37 类和对象的关系是什么 一.总结 一句话总结:类生成对象,对象是类的实例化,一定是先有类,后有对象,一定是先有标准,再有个体. 1.oop的三大优势是什么? 重用性,灵活性.扩展 ...

  6. numpy,scipy,pandas 和 matplotlib

    numpy,scipy,pandas 和 matplotlib 本文会介绍numpy,scipy,pandas 和 matplotlib 的安装,环境为Windows10. 一般情况下,如果安装了Py ...

  7. DIV+CSS学习笔记

    第十五章 定位 static静态定位(不对它的位置进行改变,在哪里就在那里) 默认值.没有定位,元素出现在正常的流中(忽略 top, bottom, left, right 或者 z-index 声明 ...

  8. set_fix_multiple_port_nets

    set_fix_multiple_port_nets   -all    -buffer_constants 加上这个命令之后 综合之后的网表就不会出现assign语句 否则会出现

  9. 【AtCoder Beginner Contest 074 D】Restoring Road Network

    [链接]h在这里写链接 [题意] 给你任意两点之间的最短路. 让你求出原图. 或者输出原图不存在. 输出原图的边长总和的最小值. [题解] floyd算法. 先在原有的矩阵上. 做一遍floyd. 如 ...

  10. [Android5.1]ActivityManagerService启动过程分析

    ActivityManagerService(简称AMS)是Android系统的关键服务之中的一个.它的主要作用例如以下: 管理系统中全部应用进程的整个生命周期 管理应用进程中的Activity.Se ...