P4187 [USACO18JAN]Stamp Painting

样例

input
3 2 2
output
6

input
6 10 5
output
190

sol:首先可以发现,对于合法的序列,只要有一串至少连续K个相同的就可以了,其他没有限制

这当然是可以dp辣

dp[i][j]表示前i位没有,当前有j个连续相同,前面没有出现连续K个相同

统计答案的时候就是∑i={K,n} dp[i][K]*Ksm(m,n-i)

转移挺容易的

dp[1][1]=m

dp[i][1]=(m-1)*∑j={1,K-1} dp[i-1][j]

dp[i][j=(2~K)] = dp[i-1][j-1]

但是就这样裸的暴力肯定是n2

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
inline ll read()
{
ll s=;
bool f=;
char ch=' ';
while(!isdigit(ch))
{
f|=(ch=='-'); ch=getchar();
}
while(isdigit(ch))
{
s=(s<<)+(s<<)+(ch^); ch=getchar();
}
return (f)?(-s):(s);
}
#define R(x) x=read()
inline void write(ll x)
{
if(x<)
{
putchar('-'); x=-x;
}
if(x<)
{
putchar(x+''); return;
}
write(x/);
putchar((x%)+'');
return;
}
#define W(x) write(x),putchar(' ')
#define Wl(x) write(x),putchar('\n')
const ll Mod=;
const int N=;
int n,m,K;
ll dp[N][N]; //dp[i]表示前i位没有,当前有j个连续相同,前面没有出现连续K个相同
inline void Ad(ll &x,ll y)
{
x+=y;
x-=(x>=Mod)?Mod:;
}
inline ll Ksm(ll x,ll y)
{
ll ans=;
while(y)
{
if(y&) ans=ans*x%Mod;
x=x*x%Mod;
y>>=;
}
return ans;
}
int main()
{
freopen("data.in","r",stdin);
freopen("baoli.out","w",stdout);
int i,j;
ll ans=;
R(n); R(m); R(K);
dp[][]=m;
for(i=;i<=n;i++)
{
for(j=;j<=(K-)&&j<=(i-);j++) Ad(dp[i][],dp[i-][j]*(m-)%Mod);
for(j=;j<=K&&j<=i;j++)
{
dp[i][j]=dp[i-][j-];
}
}
// for(i=1;i<=n;i++)
// {
// for(j=1;j<=K;j++) W(dp[i][j]);
// puts("");
// }
for(i=K;i<=n;i++) Ad(ans,dp[i][K]*Ksm(m,n-i)%Mod);
Wl(ans);
return ;
}
/*
input
3 2 2
output
6
*/

然后面临的问题就是怎么优化这个dp,容易发现其实每次除了第一位,另外都是不变的(向右移一位而已),所以只要更新第一位的值可以了

有这样两个队列

1 2 3 4 5              Head=10 Tail=14
             x 1 2 3 4                 Head=9 Tail=13

发现了吗,只要搞一个队列,每次Head-1,Tail-1就会向左移一位,那个红色的x就是要更新的值了

#include <bits/stdc++.h>
using namespace std;
typedef long long ll;
inline ll read()
{
ll s=;
bool f=;
char ch=' ';
while(!isdigit(ch))
{
f|=(ch=='-'); ch=getchar();
}
while(isdigit(ch))
{
s=(s<<)+(s<<)+(ch^); ch=getchar();
}
return (f)?(-s):(s);
}
#define R(x) x=read()
inline void write(ll x)
{
if(x<)
{
putchar('-'); x=-x;
}
if(x<)
{
putchar(x+''); return;
}
write(x/);
putchar((x%)+'');
return;
}
#define W(x) write(x),putchar(' ')
#define Wl(x) write(x),putchar('\n')
const ll Mod=;
const int N=;
int n,m,K;
ll dp[N]; //dp[j]表示前i位(已经滚掉)没有,当前有j个连续相同,前面没有出现连续K个相同
ll Queue[N<<];
ll Bin[N];
inline void Ad(ll &x,ll y)
{
x+=y;
x-=(x>=Mod)?Mod:;
x+=(x<)?Mod:;
}
int main()
{
freopen("data.in","r",stdin);
freopen("my.out","w",stdout);
int i;
ll Sum=,ans=;
R(n); R(m); R(K);
dp[]=m; for(i=;i<=K;i++) dp[i]=;
Sum=m;
for(i=;i<=K;i++) Queue[n+i]=dp[i];
Bin[]=; for(i=;i<=n;i++) Bin[i]=Bin[i-]*m%Mod;
int Head=n+,Tail=n+K;
Ad(ans,Queue[Tail]*Bin[n-]%Mod);
for(i=;i<=n;i++)
{
Queue[Head-]=(Sum-Queue[Tail]+Mod)*(m-)%Mod;
Ad(Sum,Queue[Head-]);
Ad(Sum,(-)*Queue[Tail]);
Head--;
Tail--;
// for(int j=Head;j<=Tail;j++) W(Queue[j]);
// puts("");
Ad(ans,Queue[Tail]*Bin[n-i]%Mod);
}
Wl(ans);
return ;
}
/*
input
3 2 2
output
6 input
6 10 5
output
190
*/
/*
1 2 3 4 5 Head=10 Tail=14
1 2 3 4 Head=9 Tail=13
*/

Ps:最后附上对拍

:loop
make.exe
luogu4187.exe
baoli.exe
fc my.out baoli.out
if not errorlevel goto loop
pause
goto loop

pai

#include <bits/stdc++.h>
using namespace std;
typedef int ll;
inline ll read()
{
ll s=;
bool f=;
char ch=' ';
while(!isdigit(ch))
{
f|=(ch=='-'); ch=getchar();
}
while(isdigit(ch))
{
s=(s<<)+(s<<)+(ch^); ch=getchar();
}
return (f)?(-s):(s);
}
#define R(x) x=read()
inline void write(ll x)
{
if(x<)
{
putchar('-'); x=-x;
}
if(x<)
{
putchar(x+''); return;
}
write(x/);
putchar((x%)+'');
return;
}
#define W(x) write(x),putchar(' ')
#define Wl(x) write(x),putchar('\n')
const int Mod=;
int main()
{
freopen("data.in","w",stdout);
srand(time(NULL));
int n,m,k;
n=rand()%Mod+;
m=rand()%Mod+;
k=rand()%n+;
W(n); W(m); Wl(k);
return ;
}

make

luogu4187的更多相关文章

  1. luogu4187 [USACO18JAN]Stamp Painting (dp)

    可以发现,只要存在连续k个相同的,这个情况就一定是合法情况 然而这个不太好算,我们算不存在k个相同的,然后用$m^n$把它减掉 设f[i]为前i个,没有连续k个的 显然$f[i]=m^i ,i< ...

随机推荐

  1. mysql 实现树形的遍历

    前言:关于多级别菜单栏或者权限系统中部门上下级的树形遍历,oracle中有connect by来实现,mysql没有这样的便捷途径,所以MySQL遍历数据表是我们经常会遇到的头痛问题,下面通过存储过程 ...

  2. Feature Extractor[googlenet v1]

    1 - V1 google团队在模型上,更多考虑的是实用性,也就是如何能让强大的深度学习模型能够用在嵌入式或者移动设备上.传统的想增强模型的方法无非就是深度和宽度,而如果简单的增加深度和宽度,那么带来 ...

  3. 朱晔的互联网架构实践心得S1E7:三十种架构设计模式(上)

    朱晔的互联网架构实践心得S1E7:三十种架构设计模式(上) [下载本文PDF进行阅读] 设计模式是前人通过大量的实践总结出来的一些经验总结和最佳实践.在经过多年的软件开发实践之后,回过头来去看23种设 ...

  4. (转)C#中的那些全局异常捕获

    C#中的那些全局异常捕获(原文链接:http://www.cnblogs.com/taomylife/p/4528179.html)   1.WPF全局捕获异常       public partia ...

  5. C#.NET 大型通用信息化系统集成快速开发平台 4.1 版本 - 访问记录功能改进

    当用户数据非常庞大时需要一个功能,就是统计各种账户的访问系统的情况,用户数量的各种参数需要让管理者心里有个数. 1:信息系统中有多少有效账户?可以很方便能知道具体个数,让管理者心里有个数. 2:某个公 ...

  6. win8.1系统下安装ubuntu实现双系统实践教程

    寒假闲来无事,一程序猿哥们给发了一个linux的shell编程指南,看了几张感觉不错.于是装一个试试. 没想到一装才知道了那么的问题. 下面开始: step 1: 软件准备:Ubuntu 系统镜像,这 ...

  7. git的用法步骤讲解

    1.创建全局的本地用户名 git config --global user.name "teamemory" git config --global user.email &quo ...

  8. PS 十分钟教你做出文字穿插效果

  9. 【kindle笔记】之 《恶意》-2018-4-20

    [kindle笔记]读书记录-总 在答辩和考试和各种大作业的重压以及两天后全校停电的巨大挤压中,一口气读完了恶意这本书. 这本书是我读的东野圭吾的第二本书.第一本是心心念念的<解忧杂货店> ...

  10. C#实现,C++实现,JS实现 阿拉伯数字金额转换为中文大写金额

    推荐在线编译器  ideone 1. C#实现 :带有负数处理 //把数字金额转换成中文大写数字的函数 //带有负值处理 function changeNumMoneyToChinese(money) ...