题解:P11670 [USACO25JAN] Cow Checkups S

· · 题解

本文含有二创成分。

我和乐土仙尊乃是至交好友!

你是蛊修世界中的一名蛊师,以传播大爱为己任。\ 今天你收到至交好友乐土的传信,信中称他遭具羊、星宿两名魔头的围攻,恳求你速速送来一件关键蛊材助他脱困。

事实浮冰的形成十分繁琐,这需要你洞悉一种律道蛊方的所有变式。具体而言,这道蛊方由上、下两卷组成,每一卷均由 n 个凡蛊组成,上卷的凡蛊从左向右为 a_1,a_2,\cdots,a_n,下卷对应的蛊虫为 b_1,b_2,\cdots,b_n。蛊方的契合度为 a_i=b_i 的位置 i 的个数。

形成蛊方的变式需要你恰好做一次如下操作:选择两个正整数 l,r,将 a_{l\cdots r} 翻转。\ 由于事实往往不是完美的,你只需要知道所有 \frac{n(n+1)}{2} 种变式的契合度之和,而不需要给出具体每一种方案的契合度。

附加任务

事态紧急,若是有线性复杂度的方案就再好不过了。

思路

考虑统计每一对 a_x=b_y 配对起来做出贡献的方案数。

a_xb_x 配对

有两种情况可以使原本处在相同位置的两个蛊配对:它们不处在翻转范围内(r<xx<l),或它们是翻转操作的中心(x=\frac{l+r}{2})。\ 前者分开计算翻转区间在 x 左边和右边的的方案数,具体为:

\frac{x(x-1)}{2}+\frac{(n-x)(n-x+1)}{2}=\frac{x(x-1)+(n-x)(n-x+1)}{2}

后者计算以 x 为中心的方案数即可,简单统计发现数量为 \min(x,n-x+1)

a_xb_yx\ne y)配对

考虑拆掉这个 $\min$,统计每一个 $x$ 或 $n-y+1$ 作为较小值被取了多少次。 可以发现 $x$ 作为较小值,代表的就是 **$x$ 与序列左端的距离比 $y$ 离序列的右端的距离短**,做出贡献的次数就是比 $x$ **更靠近中央**的 $y$ 使 $a_x=b_y$ 的数量。$n-y+1$ 的贡献同理。 当然 $a_y=b_x$ 也要统计。 整体来看,做法就是从中央向两边扩展,统计 $a$、$b$ 数组中已经被扩展部分中每个值的出现次数,每次加入一个 $a_x$、$b_x$,它做出贡献的次数就是在它之前另一个数组内相同的值出现的次数,每次做出贡献的量为 $\min(x,n-x+1)$。 可以参考代码理解,时间复杂度 $O(n)$。 ## 代码 ```cpp #include<bits/stdc++.h> using namespace std; int n,a[600000],b[600000],cnta[600000],cntb[600000],l,r; long long ans; int main(){ scanf("%d",&n); for(int i = 1;i <= n;i ++) scanf("%d",&a[i]); for(int i = 1;i <= n;i ++) scanf("%d",&b[i]); l = n / 2 + 1,r = n / 2;//从中央开始扩展 for(int i = 1;i <= n;i ++) if(a[i] == b[i]) ans += min(i,n - i + 1) + (1ll * i * (i - 1) + 1ll * (n - i) * (n - i + 1)) / 2;//a_i = b_i 做出的贡献 for(int i = 1;i <= (n + 1) / 2;i ++){//每次扩展往左往右都分别扩展一个长度 if(++ r > n) break;//防越界 ans += 1ll * (cntb[a[r]] + cnta[b[r]]) * (n - r + 1);//a_r 和 b_r 做出的贡献 cnta[a[r]] ++,cntb[b[r]] ++;//统计 a_r 和 b_r 出现的次数 if(-- l <= 0) break; ans += 1ll * (cntb[a[l]] + cnta[b[l]]) * l;//向左扩展,同理 cnta[a[l]] ++,cntb[b[l]] ++; } printf("%lld",ans); return 0; } ``` ## 后记 在最关键的时刻,你终于来到了乐土身边,扭转了战局。 乐土抓住你的手臂做出/bx 的表情,双眼瞪大,感动地说不出话来。 又是传播大爱的一天呢。