luogu4187
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的更多相关文章
- luogu4187 [USACO18JAN]Stamp Painting (dp)
可以发现,只要存在连续k个相同的,这个情况就一定是合法情况 然而这个不太好算,我们算不存在k个相同的,然后用$m^n$把它减掉 设f[i]为前i个,没有连续k个的 显然$f[i]=m^i ,i< ...
随机推荐
- Flink知识点
1. Flink.Storm.Sparkstreaming对比 Storm只支持流处理任务,数据是一条一条的源源不断地处理,而MapReduce.spark只支持批处理任务,spark-streami ...
- Signalr实现消息推送
一.前言 大多数系统里面好像都有获取消息的功能,但这些消息来源都不是实时的,比如你开两个浏览器,用两个不同的账号登录,用一个账号给另外一个账号发送消息,然而并不会实时收到消息,必须要自己手动F5刷新一 ...
- MySQL 数据库 初识
---------------------------------------------确定目标,认准目标,前进,克服困难,前进,克服困难,前进克服困难,前进. # # -------------- ...
- H5 id选择器
09-id选择器 迟到毁一生 早退穷三代 按时上下班 必成高富帅 <!DOCTYPE html> <html lang="en"> <head> ...
- mysql数据的导入和导出
一. mysqldump工具基本用法,不适用于大数据备份 1. 备份所有数据库: mysqldump -u root -p --all-databases > all_database_sq ...
- 使用 Drools 和 JPA & Drools show case in docker hub
使用 Drools 和 JPA 实现持续的实时数据分析https://www.ibm.com/developerworks/cn/java/j-drools5/index.html Drools - ...
- JAVAString初始化的引用问题
1 String a="Hello JAVA"; 2 3 String b=a; 4 5 System.out.println(a); 6 7 System.out.println ...
- IdentityServer4【Topic】之登出
Sign-out 登出 IdentityServer的登出就像删除认证cookie一样简单,但是为了完成一个完整的联合签名,我们必须考虑将用户从客户端应用程序中(甚至可能是上游的Identity提供者 ...
- Linux bc 命令简单学习
1. bash里面能够实现比较简单的四则运算 echo $((*)) 注意是 双括号+ $ 地址符号. 2. 但是比较复杂的 可能就难以为继了 比如不支持精度 3. 所以这里面需要使用 bc 命令来执 ...
- java回调机制——基本理解
回调(diao):往回调用,反向调用. 英文 call back.call:调用,back:返回,往返. 回调的意思就是杀个回马枪...... 回调(callback),既然是往回调用,那自然有一个正 ...