题目背景
NOIP2015 普及组 T3、深入浅出进阶1-5
题目描述
一条狭长的纸带被均匀划分出了 nnn 个格子,格子编号从 111 到 nnn。每个格子上都染了一种颜色 coloricolor_icolori 用 [1,m][1,m][1,m] 当中的一个整数表示),并且写了一个数字 numberinumber_inumberi。
编号 | 1 | 2 | 3 | 4 | 5 | 6 |
---|---|---|---|---|---|---|
颜色和数字 | 5\color{blue}{5}5 | 5\color{blue}{5}5 | 3\color{red}{3}3 | 2\color{red}{2}2 | 2\color{blue}{2}2 | 2\color{red}{2}2 |
定义一种特殊的三元组:(x,y,z)(x,y,z)(x,y,z),其中 x,y,zx,y,zx,y,z 都代表纸带上格子的编号,这里的三元组要求满足以下两个条件:
-
x,y,zx,y,zx,y,z 都是整数,x<y<z,y−x=z−yx<y<z,y-x=z-yx<y<z,y−x=z−y。
-
colorx=colorzcolor_x=color_zcolorx=colorz。
满足上述条件的三元组的分数规定为 (x+z)×(numberx+numberz)(x+z) \times (number_x+number_z)(x+z)×(numberx+numberz)。整个纸带的分数规定为所有满足条件的三元组的分数的和。这个分数可能会很大,你只要输出整个纸带的分数除以 100071000710007 所得的余数即可。
思路:
题目等价于求对于所有满足 x≡z( mod 2),colorx=colorzx \equiv z(\bmod 2),color_x = color_zx≡z(mod2),colorx=colorz 的二元组 (x,z)(x,z)(x,z) 中 (x+z)×(numberx+numberz)(x+z) \times (number_x+number_z)(x+z)×(numberx+numberz) 数值。容易想到将题目所有输入按照不同颜色和奇数偶数进行处理。
不妨让问题特殊化:假设我们当前要处理 333 个相同颜色,编号和数字依次为:
编号 | 1 | 5 | 9 |
---|---|---|---|
数字 | a1a_1a1 | a5a_5a5 | a9a_9a9 |
有答案
ans=(1+5)(a1+a5)+(1+9)(a1+a9)+(5+9)(a5+a9)=(2×1+5+9)a1+(2×5+1+9)a5+(2×9+1+5)a9ans = (1+5)(a_1 + a_5)+(1+9)(a_1+a_9)+(5+9)(a_5+a_9)= (2\times1+5+9)a_1 +(2\times 5+1+9)a_5+(2\times9+1+5)a_9ans=(1+5)(a1+a5)+(1+9)(a1+a9)+(5+9)(a5+a9)=(2×1+5+9)a1+(2×5+1+9)a5+(2×9+1+5)a9
进一步的,上述式子等于 (1+5+9)(a1+a5+a9)+(1×a1+5×a5+9×a9)(1+5+9)(a_1+a_5+a_9)+(1\times a_1 +5 \times a_5 + 9 \times a_9)(1+5+9)(a1+a5+a9)+(1×a1+5×a5+9×a9)
一般化问题,假设处理 mmm 个同色且都为奇数(偶数)的数字,编号和数字依次为
编号:x1,x2…xmx_1,x_2 \dots x_mx1,x2…xm
数字:a1,a2…ama_1,a_2 \dots a_ma1,a2…am
有 ans=(a1+a2)(x1+x2)+(a1+a3)(x1+x3)⋯=a1(x1+x2+x1+x3⋯+x1+xm)+a2⋯+am(… )ans = (a_1 + a_2)(x_1+x_2) + (a_1+a_3)(x_1+x_3)\dots=a1(x_1+x_2+x_1+x_3\dots+x_1+x_m)+a2 \dots+a_m(\dots )ans=(a1+a2)(x1+x2)+(a1+a3)(x1+x3)⋯=a1(x1+x2+x1+x3⋯+x1+xm)+a2⋯+am(…)
化简,有 ans=a1[(m−1)x1+∑i=2mxi]+a2[(m−1)x2+∑i=1mxi−x2]⋯=a1[(m−2)x1+∑i=1mxi]+a2⋯=(a1+a2+a3…am)∑i=1mxi+(m−2)∑i=1m(aixi)ans = a_1[(m-1)x_1+\sum_{i=2}^{m}{x_i}]+a_2[(m-1)x_2+\sum_{i=1}^{m}{x_i} - x_2]\dots=a_1[(m-2)x_1+\sum_{i=1}^{m}{x_i}]+a_2\dots=(a_1+a_2+a_3\dots a_m)\sum_{i=1}^{m}{x_i} + (m-2)\sum_{i=1}^{m}{(a_ix_i)}ans=a1[(m−1)x1+∑i=2mxi]+a2[(m−1)x2+∑i=1mxi−x2]⋯=a1[(m−2)x1+∑i=1mxi]+a2⋯=(a1+a2+a3…am)∑i=1mxi+(m−2)∑i=1m(aixi)
最终,可得到以下式子:
ans=∑i=1mai×∑j=1mxi+(m−2)×∑k=1m(akxk)ans = \sum_{i=1}^{m}{a_i} \times \sum_{j=1}^{m}{x_i}+(m-2) \times \sum_{k=1}^{m}{(a_kx_k)}ans=∑i=1mai×∑j=1mxi+(m−2)×∑k=1m(akxk)
注意到以上所有式子都能在输入时处理,故本题解决,算法时间复杂度 O(n+m)O(n+m)O(n+m)
代码
#include<bits/stdc++.h>
#define int long long
const int p = 10007;
using namespace std;
int n,m;
int num[100005],c[100005];
int s[100005][2],s2[100005][2],s3[100005][2];
int ans = 0;
signed main() {
scanf("%lld %lld",&n,&m);
for(int i = 1;i <= n;i++) scanf("%lld",&num[i]);
for(int i = 1;i <= n;i++) scanf("%lld",&c[i]),s3[c[i]][i % 2]++;//统计这一类的数字数量
for(int i = 1;i <= n;i++) {
s[c[i]][i % 2] += i;//统计x数列的总和
s2[c[i]][i % 2] += num[i];//统计a数列总和
if(s3[c[i]][i % 2] >= 2)ans += (s3[c[i]][i % 2] - 2) * num[i] * i;//加上所求式子后面的那一部分
ans %= p;
}
for(int i = 1;i <= m;i++) {
for(int j = 0;j <= 1;j++) {
if(s3[i][j] <= 1) continue;
ans += s[i][j] * s2[i][j];//加上所求式子前面的那一部分
ans %= p;
}
}
printf("%lld\n",ans);
return 0;
}