北京区域赛I题,Uva7676,A Boring Problem,前缀和差分

A Boring Problem

题解

其实这题不难,只要想到了前缀和差分就基本OK了.

我们要求的是第 i i i项的式子:

F ( i ) = ( a 1 + a 2 + . . . + a i ) k + ( a 2 + . . . + a i ) k + . . . + ( a i ) k F(i)=(a_1+a_2+...+a_i)^k+(a2+...+a_i)^k+...+(a_i)^k F(i)=(a1+a2+...+ai)k+(a2+...+ai)k+...+(ai)k

S i = a 1 + a 2 + . . . + a i , S 0 = 0 S_i = a_1+a_2+...+a_i,S_0=0 Si=a1+a2+...+ai,S0=0

F ( i ) = ( S i − S 0 ) k + ( S i − S 1 ) k + . . . + ( S i − S i − 1 ) k F(i)=(S_i-S_0)^k+(S_i-S_1)^k+...+(S_i-S_{i-1})^k F(i)=(SiS0)k+(SiS1)k+...+(SiSi1)k

二项式定理展开:

F ( i ) = ∑ t = 0 k C k t S i t ( − S 0 ) k − t + ∑ t = 0 k C k t S i t ( − S 1 ) k − t + . . . + ∑ t = 0 k C k t S i t ( − S i − 1 ) k − t F(i)=\sum_{t=0}^kC_k^tS_i^t(-S_0)^{k-t}+\sum_{t=0}^kC_k^tS_i^t(-S_1)^{k-t}+...+\sum_{t=0}^kC_k^tS_i^t(-S_{i-1})^{k-t} F(i)=t=0kCktSit(S0)kt+t=0kCktSit(S1)kt+...+t=0kCktSit(Si1)kt

整理得:

F ( i ) = ∑ t = 0 k C k t S i t ( − 1 ) k − t ( S 0 k − t + S 1 k − t + . . . + S i − 1 k − t ) F(i) = \sum_{t=0}^kC_k^tS_i^t(-1)^{k-t}(S_0^{k-t}+S_1^{k-t}+...+S_{i-1}^{k-t}) F(i)=t=0kCktSit(1)kt(S0kt+S1kt+...+Si1kt)

再记

S S [ i ] [ j ] = S 0 i + S 1 i + . . . + S j − 1 i SS[i][j] =S_0^i+S_1^i+...+S_{j-1}^i SS[i][j]=S0i+S1i+...+Sj1i

那么

F ( i ) = ∑ t = 0 k C k t S i t ( − 1 ) k − t ( S S [ k − t ] [ i − 1 ] ) F(i) = \sum_{t=0}^kC_k^tS_i^t(-1)^{k-t}(SS[k-t][i-1]) F(i)=t=0kCktSit(1)kt(SS[kt][i1])

注意到 S S SS SS可以 O ( n k ) O(nk) O(nk)预处理出来, S S S可以 O ( n ) O(n) O(n)预处理出来,而 F ( i ) F(i) F(i)就可以 O ( k ) O(k) O(k)得到.

注意! S 0 0 = 1 S_0^0 = 1 S00=1

代码

#include <iostream>
#include <algorithm>
#include <cstring>
#define pr(x) std::cout << #x << ':' << x << std::endl
#define rep(i,a,b) for(int i = a;i <= b;++i)

typedef long long LL;
const int N = 100010;
const LL P = 1e9+7;
int T,n,k;
char s[N];
long long S[101][N],SS[101][N];
long long C[110][110];
void init() {
	C[0][0] = 1;
	for(int i = 1;i <= 100;++i) {
		C[i][0] = 1;
		for(int j = 1;j <= i;++j) {
			C[i][j] = (C[i-1][j-1] + C[i-1][j]) % P;
		}
	}
}
int main() {
	std::ios::sync_with_stdio(false);
	init();
	std::cin >> T;
	while(T--) {
		std::cin >> n >> k;
		std::cin >> s;
		for(int i = 0;i <= n;++i) S[0][i] = 1;
		for(int i = 1;i <= n;++i) S[1][i] = (s[i-1]-'0') + S[1][i-1] ;
		for(int i = 2;i <= k;++i) 
			for(int j = 1;j <= n;++j)
				S[i][j] = S[1][j] * S[i-1][j] % P;
		
		SS[0][0] = 1;

		for(int i = 0;i <= k;++i) {
			for(int j = 1;j <= n;++j) 
				SS[i][j] = (SS[i][j-1] + S[i][j])% P;
		}
		
		for(int i = 1;i <= n;++i) {
			long long ans = 0;
			for(int j = 0;j <= k;++j) {
				long long res = C[k][j]*S[j][i]%P*SS[k-j][i-1]%P;
				if((k-j)%2==0) ans = (ans + res) % P;
				else ans = (ans - res + P) % P;
			}
			if(i != 1) std::cout << " ";
			std::cout << ans;
		}
		std::cout << std::endl;
	}
	return 0;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值