【数学】P2671 [NOIP2015 普及组] 求和

题目背景

NOIP2015 普及组 T3、深入浅出进阶1-5

题目描述

一条狭长的纸带被均匀划分出了 nnn 个格子,格子编号从 111nnn。每个格子上都染了一种颜色 coloricolor_icolori[1,m][1,m][1,m] 当中的一个整数表示),并且写了一个数字 numberinumber_inumberi

编号123456
颜色和数字5\color{blue}{5}55\color{blue}{5}53\color{red}{3}32\color{red}{2}22\color{blue}{2}22\color{red}{2}2

定义一种特殊的三元组:(x,y,z)(x,y,z)(x,y,z),其中 x,y,zx,y,zx,y,z 都代表纸带上格子的编号,这里的三元组要求满足以下两个条件:

  1. x,y,zx,y,zx,y,z 都是整数,x<y<z,y−x=z−yx<y<z,y-x=z-yx<y<z,yx=zy

  2. 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_zxz(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 个相同颜色,编号和数字依次为:

编号159
数字a1a_1a1a5a_5a5a9a_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,x2xm

数字:a1,a2…ama_1,a_2 \dots a_ma1,a2am

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[(m1)x1+i=2mxi]+a2[(m1)x2+i=1mxix2]=a1[(m2)x1+i=1mxi]+a2=(a1+a2+a3am)i=1mxi+(m2)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+(m2)×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;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值