题解 P5204 【[USACO19JAN]Train Tracking 2】
毒毒题,对着嘤文题解看了贼久
首先考虑此题的一个弱化版本:如果输入的所有
现在假设有
那么取值就只有
那么有一个显然的dp,
枚举这个数列的最后一个取值为
这个dp显然是不行的,还有一个dp,也是设
第
为什么减掉的是
il int solve(int v,int len){
int x=1000000000-v,xk=pow(x,k);
f[0]=f[1]=1;
for(int i=2;i<=len+1;++i){
f[i]=1ll*(x+1)*f[i-1]%mod;
if(i-k-1>=0)f[i]=(f[i]-1ll*xk*f[i-k-1]%mod+mod)%mod;
}
return f[len+1];
}
那么解决了前面的问题,后面的也很好办
设
首先,可以将一段相等的
然后(开始口胡)
对于一个段
如果有一个
那么可以知道的是
这里可以推出
前面已经推出
这一段的最小值只能有一个就是
前面的数也都在前一段的范围内
所以这一段最前面
对于后面也同理,如果
所以一段的len实际上是
对于所有的段依次球解最后方案数乘起来即可
代码极为简单
#include<bits/stdc++.h>
#define il inline
#define vd void
#define mod 1000000007
typedef long long ll;
il ll gi(){
ll x=0,f=1;
char ch=getchar();
while(!isdigit(ch)){
if(ch=='-')f=-1;
ch=getchar();
}
while(isdigit(ch))x=x*10+ch-'0',ch=getchar();
return x*f;
}
int n,k;
int a[100010];
int f[100010];
il int pow(int x,int y){
int ret=1;
while(y){
if(y&1)ret=1ll*ret*x%mod;
x=1ll*x*x%mod;y>>=1;
}
return ret;
}
il int solve(int v,int len){
int x=1000000000-v,xk=pow(x,k);
f[0]=f[1]=1;
for(int i=2;i<=len+1;++i){
f[i]=1ll*(x+1)*f[i-1]%mod;
if(i-k-1>=0)f[i]=(f[i]-1ll*xk*f[i-k-1]%mod+mod)%mod;
}
return f[len+1];
}
int main(){
#ifndef ONLINE_JUDGE
freopen("in.in","r",stdin);
freopen("out.out","w",stdout);
#endif
n=gi(),k=gi();
for(int i=1;i<=n-k+1;++i)a[i]=gi();
int ans=1,len;
for(int i=1,j;i<=n-k+1;i=j+1){
j=i;
while(a[j+1]==a[i])++j;
len=j-i+k;
if(i!=1&&a[i-1]>a[i])len-=k;
if(j!=n-k+1&&a[j+1]>a[i])len-=k;
if(len>0)ans=1ll*ans*solve(a[i],len)%mod;
}
printf("%d\n",ans);
return 0;
}