首页 > 其他 > 详细

LOJ 6261 一个人的高三楼

时间:2020-04-02 19:44:48      阅读:63      评论:0      收藏:0      [点我收藏+]

LINK:一个人的高三楼

求一个数列的K次前缀和。

\(n\leq 100000,k\leq 2^60\)

前缀和K次。这类似于多项式卷积。

构造 普通型生成函数F(x)表示数列中的每一项 0次前缀和F(x)=\sum_{i=0}{n-1}a_{i+1}x^i

考虑如何生成1次前缀和 构造多项式G(x).

使得 F(x)[0]×G(x)=F(x)[1].

可以比较显然的得到\(G(x)=\sum_{i=0}{n-1}x^i\)

所以所求为F(x)×(G(x))^k.

多项式快速幂 复杂度nlognlogk.常数过大过不了。

由于G(x)这个多项式的系数的特殊性 考虑O(1)求出\(G(x)^k\)的各项系数。

我们只要前n-1项 对于第i项的系数 有K个多项式的某项系数提供 可以写成 \(\sum_{i=1}{k}w_i=i\)

显然是隔板法 系数为C(i+k-1,k-1)=C(i+k-1,i).

直接求不太好求 但是 这些系数是递增的 如C(i+k-1,i),C(i+1+k-1,i+1),C(i+2+k-1,i+2)...

第一项好求为1 第二项可以由第一项推出来。

所以总复杂度 nlogn.

//#include<bits\stdc++.h>
#include<iostream>
#include<iomanip>
#include<cstdio>
#include<cstring>
#include<string>
#include<ctime>
#include<cmath>
#include<cctype>
#include<cstdlib>
#include<queue>
#include<deque>
#include<stack>
#include<vector>
#include<algorithm>
#include<utility>
#include<bitset>
#include<set>
#include<map>
#define ll long long
#define db double
#define INF 1000000000
#define ldb long double
#define pb push_back
#define get(x) x=read()
#define gt(x) scanf("%d",&x)
#define put(x) printf("%d\n",x)
#define putl(x) printf("%lld\n",x)
#define gc(a) scanf("%s",a+1)
#define rep(p,n,i) for(RE ll i=p;i<=n;++i)
#define go(x) for(ll i=lin[x],tn=ver[i];i;tn=ver[i=nex[i]])
#define fep(n,p,i) for(RE ll i=n;i>=p;--i)
#define pii pair<ll,ll> 
#define mk make_pair
#define mod 998244353
#define RE register
#define gf(x) scanf("%lf",&x)
#define pf(x) ((x)*(x))
#define ull unsigned long long
#define P 1000000000000000ll
using namespace std;
char buf[1<<15],*fs,*ft;
inline char getc()
{
	return (fs==ft&&(ft=(fs=buf)+fread(buf,1,1<<15,stdin),fs==ft))?0:*fs++;
}
inline ll read()
{
	RE ll x=0,f=1;char ch=getc();
	while(ch<‘0‘||ch>‘9‘){if(ch==‘-‘)f=-1;ch=getc();}
	while(ch>=‘0‘&&ch<=‘9‘){x=x*10+ch-‘0‘;ch=getc();}
	return x*f;
}
const ll MAXN=300010,G=3;
ll n,k,lim;
ll a[MAXN],b[MAXN],rev[MAXN];
ll ksm(ll b,ll p)
{
	b=b%mod;ll cnt=1;
	while(p)
	{
		if(p&1)cnt=cnt*b%mod;
		b=b*b%mod;p=p>>1;
	}
	return cnt;
}
inline void NTT(ll *a,ll op)
{
	rep(0,lim-1,i)if(rev[i]<i)swap(a[i],a[rev[i]]);
	for(ll len=2;len<=lim;len=len<<1)
	{
		ll mid=len>>1;
		ll wn=ksm(G,op==1?(mod-1)/len:mod-1-(mod-1)/len);
		for(ll j=0;j<lim;j+=len)
		{
			ll d=1;
			for(ll i=0;i<mid;++i)
			{
				ll x=a[i+j],y=a[i+j+mid]*d%mod;
				a[i+j]=(x+y)%mod;a[i+j+mid]=(x-y+mod)%mod;
				d=d*wn%mod;
			}
		}
	}
	if(op==-1)
	{
		ll inv=ksm(lim,mod-2);
		rep(0,lim-1,i)a[i]=a[i]*inv%mod;
	}
}
signed main()
{
	freopen("1.in","r",stdin);
	get(n);get(k);b[0]=1;
	rep(0,n-1,i)get(a[i]);
	if(!k){rep(0,n-1,i)putl(a[i]);return 0;}
	rep(1,n-1,i)b[i]=(b[i-1]+b[i-1]*ksm(i,mod-2)%mod*((k-1)%mod)%mod)%mod;
	//b[i]=C(i+k-1,i)=C(i+k-2,i-1)+C(i+k-2,i);
	lim=1;while(lim<n+n)lim=lim<<1;
	rep(1,lim-1,i)rev[i]=rev[i>>1]>>1|((i&1)?lim>>1:0);
	NTT(a,1);NTT(b,1);
	rep(0,lim-1,i)a[i]=a[i]*b[i]%mod;
	NTT(a,-1);
	rep(0,n-1,i)putl(a[i]);
	return 0;
}

NTT好久没打了 所以调了1h.

LOJ 6261 一个人的高三楼

原文:https://www.cnblogs.com/chdy/p/12622403.html

(0)
(0)
   
举报
评论 一句话评论(0
关于我们 - 联系我们 - 留言反馈 - 联系我们:wmxa8@hotmail.com
© 2014 bubuko.com 版权所有
打开技术之扣,分享程序人生!