C - Swaps 2
给定两个长度为 n n n的数组 A , B A, B A,B,我们可以进行若干次如下操作,使 A A A变成 B B B,
- 选定 i < n i < n i<n,将 a i a_i ai减小 1 1 1,将 a i + 1 a_{i + 1} ai+1增加 1 1 1。
- 交换 a i , a i + 1 a_i, a_{i + 1} ai,ai+1。
问我们进行多少次操作,可以将 A A A变成 B B B,如果无解则输出 − 1 -1 −1,否则输出最小操作次数。
容易发现,不管怎么交换 a i + i a_i + i ai+i的值都是不变的,当我们要得到 b i b_i bi的时候,我们一定是去找一个 a j + j = b i + i a_j + j = b_i + i aj+j=bi+i,且 j j j最小的那个点,这样可以减小操作次数,
#include <bits/stdc++.h>
using namespace std;
const int N = 2e5 + 10;
int a[N], b[N], c[N], sum[N], n, m;
vector<int> vt[N];
inline int lowbit(int x) {
return x & (-x);
}
void update(int x, int v) {
while (x <= n) {
sum[x] += v;
x += lowbit(x);
}
}
int query(int x) {
int ans = 0;
while (x) {
ans += sum[x];
x -= lowbit(x);
}
return ans;
}
int query(int l, int r) {
return query(r) - query(l - 1);
}
int main() {
// freopen("in.txt", "r", stdin);
// freopen("out.txt", "w", stdout);
scanf("%d", &n);
for (int i = 1; i <= n; i++) {
scanf("%d", &a[i]);
c[i] = a[i] + i;
}
sort(c + 1, c + 1 + n);
m = unique(c + 1, c + 1 + n) - (c + 1);
for (int i = 1; i <= n; i++) {
scanf("%d", &b[i]);
int pos = lower_bound(c + 1, c + 1 + m, b[i] + i) - c;
if (c[pos] != b[i] + i) {
puts("-1");
return 0;
}
b[i] = pos;
vt[lower_bound(c + 1, c + 1 + m, a[i] + i) - c].push_back(i);
}
long long ans = 0;
for (int i = 1; i <= n; i++) {
if (!vt[b[i]].size()) {
puts("-1");
return 0;
}
int cur = vt[b[i]].back();
ans += cur + query(cur + 1, n) - i;
vt[b[i]].pop_back();
update(cur, 1);
}
printf("%lld\n", ans);
return 0;
}