首页 > 其他 > 详细

CF 1416 C.XOR Inverse【字典树求逆序对】

时间:2021-04-28 22:25:34      阅读:66      评论:0      收藏:0      [点我收藏+]

XOR Inverse
思路
将一个数二进制拆位,考虑x和y谁大谁小即看第一个不相同的二进制位上两个数的大小即可。因此贪心的从高位到低位考虑,比较每一位剩余所有数<0,1>对或者<1,0>对的数量的多少判断这一位是否要对所有的数进行更改。
主要是讲述字典树求逆序对的方法:
1、将所有数插入到字典树中,并将每个数的下标插入到其所对应的每一个0,1路径上。
2、最关键的一部分:抓住性质,比较两个数大小是比较两个数第一个不相同的二进制位。那么对于一个结点来说,其左节点的数一定都是小于右节点的数。因此遍历一遍左节点中所存入的下标,同时拿另一个指针移动右节点中存入的下标。当且仅当左节点当前下标>右节点下标时,就有逆序对贡献。
代码

#include<bits/stdc++.h>
using namespace std;

typedef long long LL;
typedef pair<int, int> PII;
const int N = 3e5 + 10;
const int inf = 0x3f3f3f3f;
const int mod = 1e9 + 7;

int tr[N * 30][2], idx;
vector<int> vec[N * 30];
LL f[31][2];
int a[N];

void insert(int val, int id) {
	int p = 0;
	for(int i = 30; i >= 0; i--) {
		int x = (val >> i) & 1;
		if(!tr[p][x]) tr[p][x] = ++idx;
		p = tr[p][x];
		vec[p].push_back(id);
	}
}

void dfs(int dep, int p) {
	if(dep < 0) return;
	int l = tr[p][0], r = tr[p][1];
	LL cnt = 0, idx = 0;
	LL ls = vec[l].size(), rs = vec[r].size();
	for(int i = 0; i < ls; i++) {
		while(idx < rs && vec[l][i] > vec[r][idx]) idx++;
		cnt += idx;
	}
	f[dep][0] += cnt;
	f[dep][1] += 1LL * ls * rs - cnt;
	if(tr[p][0]) dfs(dep - 1, tr[p][0]);
	if(tr[p][1]) dfs(dep - 1, tr[p][1]);
}

void solve() {
	int n;
	scanf("%d", &n);
	for(int i = 1; i <= n; i++) {
		scanf("%d", &a[i]);
		insert(a[i], i);
	}
	dfs(30, 0);
	int num = 0;
	LL res = 0;
	for(int i = 30; i >= 0; i--) {
		if(f[i][0] <= f[i][1]) {
			res += f[i][0];
		}
		else {
			res += f[i][1];
			num += (1 << i);
		}
	}
	printf("%lld %d\n", res, num);
}

int main() {
    // freopen("in.txt", "r", stdin);
    solve();
    return 0;
}

CF 1416 C.XOR Inverse【字典树求逆序对】

原文:https://www.cnblogs.com/ZX-GO/p/14715753.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!