逆数对(树状数组的方法)

本文介绍了如何使用树状数组算法解决计算给定整数序列中逆序对总数的问题,通过单点添加更新和区间查询来快速求解,时间复杂度为O(logn)。
摘要由CSDN通过智能技术生成

本题链接:登录—专业IT笔试面试备考平台_牛客网

题目:

样例:

输入
5
4 5 1 3 2

输出
7

思路:

        根据题意,求逆序对总数。

        逆序对含义:如果数组中的两个不同位置,前面的数字比后面的数字严格大,则称其为一个逆序对。

        根据样例已知: 4 5 1 3 2

        我们可以通过计数的方式,log(n)的时间复杂度获取逆序对。

        方式如下:

        每输入一个 x 后

        

00000
12...45

        就开始询问有多少个逆序对,求总和(x + 1 ~ INF) 的数量是多少。

        比如当输入到 1 的时候:

        

10011
12345

        逆序对的数量为:  sum(2 ~ INF) = 0 + 0 + 1 + 1 + 0 + ... + 0 = 2

        依此类推。

        这里利用到了树状数组(单点添加,区间查询)。操作函数看我以往的笔记。

        代码详解如下:

#include <iostream>
#include <vector>
#include <queue>
#include <cstring>
#include <algorithm>
#include <unordered_map>
#define endl '\n'
#define int long long
#define YES puts("YES")
#define NO puts("NO")
#define umap unordered_map
#define All(x) x.begin(),x.end()
#pragma GCC optimize(3,"Ofast","inline")
#define IOS std::ios::sync_with_stdio(false),cin.tie(0), cout.tie(0)
using namespace std;
const int N = 2e6 + 10;
inline void solve();

signed main()
{
//	freopen("a.txt", "r", stdin);
	IOS;
	int _t = 1;
	// cin >> _t;
	while (_t--)
	{
		solve();
	}
	return 0;
}
int R = 100010;	// 题目 a[i] 的临界范围
int cnt[N];	// 用于计数
inline int lowbit(int x){return x&(-x);}
// 单点添加函数
inline void Add_pos(int pos,int x)
{
	for(int i = pos;i <= R;i += lowbit(i)) cnt[i] += x; 
}
// 区间询问函数	变相的获取逆序对
inline int ask(int l,int r)
{
	int ans = 0;
	for(int i = l - 1;i;i-=lowbit(i)) ans -= cnt[i];
	for(int i = r;i;i-=lowbit(i)) ans += cnt[i];
	return ans;
}
inline void solve()
{
	int n,x,ans = 0;
	cin >> n;
	while(n--)
	{
		cin >> x;
		Add_pos(x,1); // 单点添加,统计
		ans += ask(x + 1,R);	// 询问总和,获取逆序对数量
	}
	cout << ans << endl;
}

  最后提交:

  • 5
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值