F.Easy Fix
题目分析
给定排列p 1 , p 2 , p 3 , … , p n p_1, p_2, p_3,\dots, p_np1,p2,p3,…,pn,定义A i A_iAi表示在p i p_ipi左侧并比p i p_ipi小的数字个数,B i B_iBi表示在p i p_ipi右侧并比p i p_ipi小的数字个数,C i = min ( A i , B i ) C_i = \min(A_i, B_i)Ci=min(Ai,Bi)。现在给定多个操作( l , r ) (l, r)(l,r),求每个操作,交换( p i , p j ) (p_i, p_j)(pi,pj)后的∑ C i \sum C_i∑Ci。
首先考虑如何处理初始时的C i C_iCi值,观察到以下性质:
- 对于A i A_iAi值的求解过程类似求逆序对的思想,可以直接上树状数组维护,O ( n log n ) O(n\log n)O(nlogn)求得全部的A i A_iAi
- 由于是排列,B i = p i − 1 − A i B_i = p_i - 1 - A_iBi=pi−1−Ai可以O ( 1 ) O(1)O(1)求得
- 那么C i = min ( A i , B i ) C_i = \min(A_i, B_i)Ci=min(Ai,Bi)也是O ( 1 ) O(1)O(1)得到的
由于每个询问相互独立,那么考虑交换( p l , p r ) (p_l, p_r)(pl,pr)操作对C i C_iCi的影响:
对于[ 1 , l ) , ( r , n ] [1, l), (r , n][1,l),(r,n]范围的数字,C i C_iCi值一定不影响。因为交换操作均在单侧进行
对于p l p_lpl,交换到r rr位置后,A l → A l + 区 间 [ l , r ] 小 于 p l 的 数 字 个 数 A_l \rightarrow A_l + 区间[l,r]小于p_l的数字个数Al→Al+区间[l,r]小于pl的数字个数,B l ’ B_l’Bl’仍然可以直接求
对于p r p_rpr,交换到l ll位置后,A r → A r − 区 间 [ l , r ] 小 于 p r 的 数 字 个 数 A_r \rightarrow A_r - 区间[l ,r]小于p_r的数字个数Ar→Ar−区间[l,r]小于pr的数字个数,B r ’ B_r’Br’仍然可以直接求
如果我们在线询问(主席树维护),那么对于p l , p r p_l,p_rpl,pr,实际上可以直接两个O ( l o g n ) O(logn)O(logn)重新求。
那么重点是对于[ l + 1 , r − 1 ] [l + 1, r - 1][l+1,r−1]区间内的数字的C i C_iCi值变化,如何维护?
对于p l ≤ p i ≤ p r p_l \leq p_i \leq p_rpl≤pi≤pr,如果A i ≤ B i A_i \leq B_iAi≤Bi,则交换后A i − 1 , B i + 1 A_i - 1, B_i + 1Ai−1,Bi+1,从而C i − 1 C_i - 1Ci−1
对于p l ≥ p i ≥ p r p_l \geq p_i \geq p_rpl≥pi≥pr,如果A i ≥ B i A_i \geq B_iAi≥Bi,则交换后A i + 1 , B i − 1 A_i + 1, B_i - 1Ai+1,Bi−1,从而C i − 1 C_i -1Ci−1
对于p l ≤ p i ≤ p r p_l \leq p_i \leq p_rpl≤pi≤pr,如果A i − 1 ≥ B i + 1 , A i ≥ B i A_i - 1 \geq B_i + 1, A_i \geq B_iAi−1≥Bi+1,Ai≥Bi,则交换后C i + 1 C_i + 1Ci+1
对于p l ≥ p i ≥ p r p_l \geq p_i \geq p_rpl≥pi≥pr,如果A i − 1 ≤ B i + 1 , A i ≤ B i A_i - 1 \leq B_i + 1, A_i \leq B_iAi−1≤Bi+1,Ai≤Bi,则交换后C i + 1 C_i + 1Ci+1
那么对于以上四种情况,我们可以分别用四棵主席树进行维护。同时,对于p l , p r p_l, p_rpl,pr的贡献计算还需要支持区间< K <K<K的数字个数查询,因此共需五棵主席树进行维护,复杂度O ( m × 4 log n ) O(m \times 4 \log n)O(m×4logn)。
Code
#include <bits/stdc++.h>
#define int long long
#define endl '\n'
const int N = 1e5 + 10;
int p[N], a[N], b[N], c[N], d[N];
using bset5 = std::bitset<5>;
namespace Fenwick{
int tree[N], len;
#define lowbit(x) ((x) & (-x))
inline void init(int ln){ len = ln; }
inline void update(int i, int x){ for(int pos = i; pos <= len; pos += lowbit(pos)) tree[pos] += x; }
inline int getsum(int i, int ans = 0){ for(int pos = i; pos; pos -= lowbit(pos)) ans += tree[pos]; return ans; }
}
namespace PresidentTree{
int root[N], sum[N << 5][5], lc[N << 5], rc[N << 5], cnt;
#define ls l, mid
#define rs mid + 1, r
void update(int &rt, int pre, int l, int r, int x, bset5 inc){
rt = ++cnt, lc[rt] = lc[pre], rc[rt] = rc[pre];
for(int i = 0; i <= 5; i++) sum[rt][i] = sum[pre][i] + (inc[i] ? 1 : 0);
if(l == r) return;
int mid = l + r >> 1;
if(x <= mid) update(lc[rt], lc[rt], l, mid, x, inc);
else update(rc[rt], rc[rt], mid + 1, r, x, inc);
}
int query(int st, int ed, int l, int r, int L, int R, int id){
if(l == L && r == R) return sum[ed][id] - sum[st][id];
int mid = l + r >> 1;
if(mid >= R) return query(lc[st], lc[ed], l, mid, L, R, id);
else if(mid >= L) return query(lc[st], lc[ed], l, mid, L, mid, id) + query(rc[st], rc[ed], mid + 1, r, mid + 1, R, id);
else return query(rc[st], rc[ed], mid + 1, r, L, R, id);
}
}
#define Pdt PresidentTree
inline void solve(){
int n = 0; std::cin >> n;
Fenwick::init(n);
for(int i = 1; i <= n; i++) std::cin >> p[i];
for(int i = 1; i <= n; i++){
a[i] = Fenwick::getsum(p[i]);
b[i]= p[i] - 1 - a[i];
Fenwick::update(p[i], 1);
c[i] = std::min(a[i], b[i]);
d[i] = d[i - 1] + c[i];
}
for(int i = 1; i <= n; i++){
bset5 flag; flag.reset();
if(a[i] <= b[i]) flag[1] = true;
if(a[i] >= b[i]) flag[3] = true;
if(a[i] - 1 >= b[i] + 1 && a[i] >= b[i]) flag[2] = true;
if(a[i] + 1 <= b[i] - 1 && a[i] <= b[i]) flag[4] = true;
flag[0] = true;
Pdt::update(Pdt::root[i], Pdt::root[i - 1], 1, n + 1, p[i], flag);
}
int m = 0; std::cin >> m;
while(m--){
int l, r; std::cin >> l >> r;
if(l == r){ std::cout << d[n] << endl; continue; }
else if(l > r) std::swap(l, r);
int ans = d[n] - c[l] - c[r];
if(p[l] < p[r]){
ans -= Pdt::query(Pdt::root[l], Pdt::root[r - 1], 1, n + 1, p[l], p[r], 1)
- Pdt::query(Pdt::root[l], Pdt::root[r - 1], 1, n + 1, p[l], p[r], 2);
} else {
ans -= Pdt::query(Pdt::root[l], Pdt::root[r - 1], 1, n + 1, p[r], p[l], 3)
- Pdt::query(Pdt::root[l], Pdt::root[r - 1], 1, n + 1, p[r], p[l], 4);
}
int nowa = Pdt::query(Pdt::root[0], Pdt::root[l - 1], 1, n + 1, 1, p[r], 0),
nowb = p[r] - 1 - nowa;
ans += std::min(nowa, nowb);
nowa = Pdt::query(Pdt::root[r], Pdt::root[n], 1, n + 1, 1, p[l], 0),
nowb = p[l] - 1 - nowa;
ans += std::min(nowa, nowb);
std::cout << ans << endl;
}
}
signed main(){
std::ios_base::sync_with_stdio(false), std::cin.tie(0);
solve();
return 0;
}