题目大意:

  设$S(n,m)$为第二类斯特林数,$F_i$表示斐波那契数列第$i$项。

  给定$n,R,K$,求$\sum\limits_{i=1}^{n}(\sum\limits_{m=1}^{R}F_i)!i!\sum\limits_{l=0}^{i}\sum\limits_{j=0}^{\sum\limits_{t=1}^{R}F_t}\frac{S(k,i-l)}{l!}\frac{S(i,\sum\limits_{w=1}^{R}F_w-j)}{j!}$的值$mod$ $1000000007$。

  其中$n,R\leq10^{18}$,$k\leq2*10^5$

题解:

  (爽感)

  这道题...精神污染。。。当然也包括题解的$O(k)$做法。。。

  于是我就自己YY了一个$O(klogk)$的能过的做法.

  首先,需要知道三个小结论:

    1.$\sum\limits_{i=1}^{n}F_i=F_{n+2}-1$,用归纳法可以轻松证明。

    2.$m^n=\sum\limits_{i=1}^{m}S(n,i)C_m^ii!$

    3.$S(n,m)=\frac{1}{m!}\sum\limits_{i=0}^m(-1)^iC_m^i(m-i)^n$,本质上为结论2经二项式反演后的结果。

  证明一下第二个结论:

    考虑构造,把$n$个不同的球放入$m$个不同的盒子里,允许有空盒的方案数。

    显然,每个球都有$m$种选择方法,答案是$m^n$。

    换一种思维方式,可以想成枚举这$n$个球放入了那些盒子里,由于$S(n,m)$表示$n$个不同的球放入$m$个相同的盒子里无空盒的方案数,所以在乘上$C_m^ii!$。

  然后就可以开心的化式子了:

  设$L=\sum\limits_{m=1}^{R}F_i$,再把原式挪一下就变成了:

    $\sum\limits_{i=1}^{n}\sum\limits_{l=0}^{i}i!\frac{S(k,i-l)}{l!}\sum\limits_{j=0}^LL!\frac{S(i,L-j)}{j!}$

  然后我们神奇的发现后面的两个式子形式相同,以后一个为例:

    $\sum\limits_{j=0}^LL!\frac{S(i,L-j)}{j!}$,换成枚举$L-j$

    $=\sum\limits_{j=0}^LL!\frac{S(i,j)}{(L-j)!}$

    $=\sum\limits_{j=0}^LS(i,j)C_L^jj!$,这不就是结论2吗,直接等于$L^i$

  因此原式直接变为$\sum\limits_{i=0}^nL^ii^k$,但$n$依然很大。

  注意当$L=1$时式子变为$\sum\limits_{i=0}^ni^k$,需要用拉格朗日插值法求解,详见CF622F.

  若$L\neq1$,我的做法是把$i^k$变回斯特林数,即$\sum\limits_{i=0}^nL^i\sum\limits_{j=0}^kS(k,j)C_i^jj!$

  交换求和号,即$\sum\limits_{j=0}^kS(k,j)j!\sum\limits_{i=j}^nC_i^jL^i$,考虑如何快速求出$\sum\limits_{i=j}^nC_i^jL^i$。

  设$g(x)=\sum\limits_{i=x}^nC_i^xL^i$

  推导一波:

  $g(x)=\sum\limits_{i=x}^nL^iC_i^x$
  $=\sum\limits_{i=x}^nL^iC_{i-1}^{x-1}+\sum\limits_{i=x}^nL^iC_{i-1}^x$
  $=L\sum\limits_{i=x-1}^{n-1}L^{i}C_{i}^{x-1}+L\sum\limits_{i=x}^{n-1}L^{i}C_{i}^x$
  $=L(g(x-1)-L^nC_n^{x-1})+L(g(x)-L^nC_n^x)$

  在整理一下就是:$g(x)=\frac{Lg(x-1)-L^{n+1}C_{n+1}^x}{1-L}$,那么就可以$O(k)$的复杂度求出$g(x)$

  到此为止,原式变为$\sum\limits_{j=1}^kS(k,j)j!g(j)$,问题就剩下如何求第二类斯特林数$S(k,j)$了。

  应用第三个结论:$S(n,m)=\frac{1}{m!}\sum\limits_{i=1}^m(-1)^iC_m^i(m-i)^n$,将其化为卷积形式:

  $S(n,m)=\sum\limits_{i=1}^m\frac{(-1)^i}{i!}\frac{(m-i)^n}{(m-i)!}$$NTT$即可。

  又由于模数不是$NTT$模数,因此需要$MTT$来实现。

 

#include<cmath>
#include<cstdio>
#include<algorithm>
#define N 530010
#define Re register
#define ld long double
#define mod 1000000007
#define pi (ld)acos(-1)
#pragma GCC optimize(3)
#pragma GCC optimize("inline")
#pragma GCC optimize("Ofast")
using namespace std;
typedef long long ll;
struct node{ll a[][];}tt,an;
int k,R[N];
ll n,r,ans,y[N],S[N],fac[N],inv[N];
node mul(const node&x,const node&y)
{
node ret;
for(Re int i = ;i<=;i++)
{
for(Re int j = ;j<=;j++)
{
ret.a[i][j] = ;
for(Re int k = ;k<=;k++)
ret.a[i][j] = (ret.a[i][j]+x.a[i][k]*y.a[k][j]%mod)%mod;
}
}return ret;
}
inline void mpow(ll y)
{
an.a[][] = an.a[][] = ;
while(y)
{
if(y&)an = mul(an,tt);
tt = mul(tt,tt),y>>=;
}
}
ll ksm(ll x,ll y)
{
ll fin = ;
x%=mod,y%=mod-;
while(y)
{
if(y&)fin = fin*x%mod;
x = x*x%mod,y>>=;
}return fin;
}
struct fs
{
ld x,y;
friend fs operator+(const fs&a,const fs&b){return (fs){a.x+b.x,a.y+b.y};}
friend fs operator-(const fs&a,const fs&b){return (fs){a.x-b.x,a.y-b.y};}
friend fs operator*(const fs&a,const fs&b){return (fs){a.x*b.x-a.y*b.y,a.x*b.y+a.y*b.x};}
}a[N],b[N],c[N],d[N],e[N],f[N],g[N],h[N];
inline void FFT(fs*A,int lim,int fl)
{
for(Re int i = ;i<lim;i++)if(i<R[i])swap(A[i],A[R[i]]);
for(Re int i = ;i<=lim;i<<=)
{
fs wn = (fs){cos(*pi/i),fl*sin(*pi/i)};
for(Re int j = ;j<lim;j+=i)
{
fs w = (fs){,};
for(Re int k = ;k<i>>;k++)
{
fs x = A[j+k],y = w*A[j+k+(i>>)];
A[j+k] = x+y,A[j+k+(i>>)] = x-y;
w = w*wn;
}
}
}if(fl==-)for(Re int i = ;i<lim;i++)A[i].x/=lim;
}
inline void p2()
{
for(Re int i = ;i<=k+;i++)y[i] = (y[i-]+ksm(i,k))%mod;
for(Re int i = ;i<=k+;i++)
{
int t = ;
for(Re int j = ;j<=k+;j++)
{
if(i==j)continue;
t = t*(n%mod-j)%mod*ksm(i-j,mod-)%mod;
}ans = (ans+y[i]*t%mod)%mod;
}printf("%lld",(ans+mod)%mod),exit();
}
int main()
{
freopen("c.in","r",stdin);
freopen("c.out","w",stdout);
scanf("%lld%lld%d",&n,&r,&k);
tt.a[][] = tt.a[][] = tt.a[][] = ;
mpow(r+);
r = an.a[][]-;
if(r==)p2();
fac[] = fac[] = inv[] = inv[] = ;
for(Re int i = ;i<=k;i++)fac[i] = fac[i-]*i%mod,inv[i] = (mod-mod/i)*inv[mod%i]%mod;
for(Re int i = ;i<=k;i++)inv[i] = inv[i]*inv[i-]%mod;
for(Re int i = ;i<=k;i++)
{
ll A = (inv[i]*(i&?-:)+mod)%mod;
ll B = ksm(i,k)*inv[i]%mod;
a[i].x = (ld)(A>>);
b[i].x = (ld)(A&0x7fff);
c[i].x = (ld)(B>>);
d[i].x = (ld)(B&0x7fff);
}
int lim = ,bit = ;
while(lim<=k<<)lim<<=,bit++;
for(Re int i = ;i<lim;i++)R[i] = (R[i>>]>>)|((i&)<<bit-);
FFT(a,lim,),FFT(b,lim,),FFT(c,lim,),FFT(d,lim,);
for(Re int i = ;i<lim;i++)
{
e[i] = a[i]*c[i];
f[i] = a[i]*d[i]+b[i]*c[i];
g[i] = b[i]*d[i];
}
FFT(e,lim,-),FFT(f,lim,-),FFT(g,lim,-);
for(Re int i = ;i<=k;i++)
{
ll E = (ll)round(e[i].x)%mod,F = (ll)round(f[i].x)%mod,G = (ll)round(g[i].x)%mod;
S[i] = (((E<<)%mod+(F<<)%mod+G)%mod+mod)%mod;
}
ll lst = (ksm(r,n+)-)*ksm(r-,mod-)%mod,now = ,t = ksm(r,n+),tt = ksm(-r,mod-);
for(Re int i = ;i<=k;i++)
{
now = now*(n%mod+-i)%mod;
lst = (lst*r%mod-now*inv[i]%mod*t%mod)%mod*tt%mod;
ans = (ans+S[i]*fac[i]%mod*lst%mod)%mod;
}printf("%lld",(ans+mod)%mod);
return ;
}

最新文章

  1. 【JVM】JVM系列之Class文件(三)
  2. C# 基于json通讯中的中文的处理
  3. CSS中!important的优先级
  4. mysql 可能会用到的一些 函数
  5. Linux大文件已删除,但df查看已使用的空间并未减少解决
  6. 技术分享:逆向分析ATM分离器
  7. YTU 2616: A代码完善--简易二元运算
  8. vim乱码处理
  9. h5 如何打包apk
  10. win32下利用python操作printer
  11. 在maven仓库中查找jar
  12. JBPM4 安装和配置
  13. ACdream 1732
  14. IOS开发创建开发证书及发布App应用(六)——打包应用
  15. ConcurrentHashMap、HashTable、HashMap的区别
  16. eclipse hadoop1.2.0配置及wordcount运行
  17. Vmware安装Ubuntu ==&gt; 连网成功
  18. python 13 常用模块 一
  19. 20165223 实验四 Android开发基础
  20. mycat中schema.xml的一些解释

热门文章

  1. How to: Display a List of Non-Persistent Objects in a Popup Dialog 如何:在弹出对话框中显示非持久化对象列表
  2. forEach和map的区别,简单写了IE低版本的原形封装
  3. SAP QM 检验批里某检验特性的取样数量跟检验计划设置不符?
  4. 设置自动获取IP和DNS
  5. C# List、Array、Dictionary之间相互转换
  6. [PHP] 编译安装swoole
  7. acwing 849 Dijkstra求最短路 I 模板
  8. java之Set接口(单列集合)
  9. [译]Vulkan教程(22)创建顶点buffer
  10. Java变量在内存中的存储