====== Atcoder Rugular Contest 126 ======
[[https://atcoder.jp/contests/arc126|比赛链接]]
===== D - Pure Straight =====
==== 题意 ====
给定长度为 $n$ 的序列,交换任意两个相邻元素的费用为 $1$。
每个元素的取值范围为 $[1\sim k]$,问使得序列的一个连续子序列恰好为 $1,2\cdots k$ 的最小费用。
==== 题解 ====
首先假设固定所选元素的初始下标为 $p_1\lt p_2\lt \cdots \lt p_k$,且最终这 $k$ 个元素的下标 $[x,x+k-1]$。
不妨记权值等于 $i$ 的元素的位置为 $b_i$,同时设 $c_i=x+i-1$。
先证明该情况下的最小费用为 $\sum_{i=1}^k |p_i-c_i|+\text{inv}(b_1,b_2\cdots,b_k)$。其中 $\text{inv}B$ 表示序列 $B$ 中的逆序对。
首先证明答案的下界为 $\sum_{i=1}^k |p_i-c_i|+\text{inv}(b_1,b_2\cdots,b_k)$。
定义 $f(P)=\sum_{i=1}^k |p_i-c_i|+\text{inv}(b_1,b_2\cdots,b_k)$ 为序列 $P$ 的势能(一定要保证 $p_1\lt p_2\lt \cdots \lt p_k$)。
如果某次交换的两个元素都属于所选元素,则 $P$ 不变,$\text{inv}B$ 至多减少 $1$。
如果某次交换的至多有一个元素都属于所选元素,则 $\sum_{i=1}^k |p_i-c_i|$ 至多减 $1$,$\text{inv}B$ 不变。
接下来构造可以取到下界的方案,其实只要先在不交换相对位置的情况下将所有元素移到 $[x,x+k-1]$ 然后再调整次序消除逆序对即可。
接下来考虑固定所选元素的初始下标为 $p_1\lt p_2\lt \cdots \lt p_k$,如何确认最优的最终下标 $[x,x+k-1]$。
由于此时 $\text{inv}(b_1,b_2\cdots,b_k)$ 是定值,所以只需要考虑 $\sum_{i=1}^k |p_i-c_i|$。记 $m=\lceil \frac k2\rceil$,不难发现 $x_m=c_m$ 时最优。此时有
$$
\sum_{i=1}^k |p_i-c_i|=\sum_{i=1}^m (c_i-p_i)+\sum_{i=m+1}^k (p_i-c_i)=\sum_{i=1}^m (p_m-p_i-(m-i))+\sum_{i=m+1}^k (p_i-p_m-(i-m))
$$
上式中 $i\lt m$ 的 $p_i$ 系数为 $1$,$i\gt m$ 的 $p_i$ 系数为 $-1$,而 $p_m$ 系数根据 $k$ 的奇偶性为 $0$ 或 $-1$。
其他项是关于 $m$ 的多项式,为定值。因此在 $\text{dp}$ 过程中维护上式式子的结果和逆序对即可。时间复杂度 $O\left(n2^k\right)$。
const int MAXN=205,MAXK=17,inf=1e9;
int a[MAXN],dp[1<=0;i--){
_for(j,0,s){
if(j&(1<=m)
v+=i;
else if(bt[0][j]+1==m){
if(k%2==0)
v-=i;
}
else
v-=i;
dp[j|(1<
===== E - Infinite Operations =====
==== 题意 ====
给定一个序列 $A$,定义一次操作为选定两个 $a_i,a_j(a_i\gt a_j)$,将两个数赋值为 $a_i-x,a_j+x$,其中 $0\lt x\le \frac {a_i-a_j}2$,并得到 $x$ 分。
定义 $F(A)$ 表示序列 $A$ 经过无限次操作后能得到的最大得分。接下来若干次单点修改操作,每次操作后求 $F(A)$。
==== 题解 ====
定义势能函数 $G(A)=\sum_{i=1}^n\sum_{j=1}^{i}|a_i-a_j|$,不难发现一次操作得 $x$ 分至少使得 $G(A)$ 减少 $2x$。
同时如果假定 $a_1\le a_2\le\cdots a_n$,则选定相邻两个值操作恰好可以得 $x$ 分同时使得 $G(A)$ 减少 $x$。
任意取两个相邻数取平均,易知经过无数次操作后有 $G(A)=0$。因此答案为 $\frac {G(A)}2$,树状数组维护答案即可,时间复杂度 $O(n\log n)$。
const int MAXN=3e5+5,MAXV=6e5+5,mod=998244353,inv2=(mod+1)/2;
int a[MAXN],b[MAXV];
pair opt[MAXN];
LL c1[MAXV],c2[MAXV];
#define lowbit(x) ((x)&(-x))
void update(int pos,int v1,int v2){
while(pos query(int pos){
pair ans=make_pair(0LL,0LL);
while(pos){
ans.first+=c1[pos];
ans.second+=c2[pos];
pos-=lowbit(pos);
}
return ans;
}
int main()
{
int n=read_int(),q=read_int();
_rep(i,1,n)
a[i]=b[i]=read_int();
_rep(i,1,q){
opt[i].first=read_int();
opt[i].second=read_int();
b[n+i]=opt[i].second;
}
sort(b+1,b+n+q+1);
int m=unique(b+1,b+n+q+1)-b;
LL ans=0,sum=0;
_rep(i,1,n){
int v=lower_bound(b+1,b+m,a[i])-b;
update(v,b[v],1);
sum+=b[v];
pair t=query(v);
ans+=1LL*b[v]*t.second-t.first+sum-t.first-1LL*b[v]*(i-t.second);
ans%=mod;
}
_rep(i,1,q){
int p=opt[i].first;
int v=lower_bound(b+1,b+m,a[p])-b;
pair t=query(v);
ans-=1LL*b[v]*t.second-t.first+sum-t.first-1LL*b[v]*(n-t.second);
ans%=mod;
update(v,-b[v],-1);
sum-=b[v];
a[p]=opt[i].second;
v=lower_bound(b+1,b+m,a[p])-b;
update(v,b[v],1);
sum+=b[v];
t=query(v);
ans+=1LL*b[v]*t.second-t.first+sum-t.first-1LL*b[v]*(n-t.second);
ans%=mod;
enter(1LL*(ans+mod)*inv2%mod);
}
return 0;
}