时间限制:C/C++ 1秒,其他语言2秒
空间限制:C/C++ 262144K,其他语言524288K
64bit IO Format: %lld
题目描述
有一个长度为 n 的 01 串,你可以每次将相邻的 k 个字符合并,得到一个新的字符并获得一定分数。得到的新字符和分数由这 k
个字符确定。你需要求出你能获得的最大分数。 输入描述: 第一行两个整数n,k。 接下来一行长度为n的01串,表示初始串。
接下来2k行,每行一个字符ci和一个整数wi,ci
表示长度为k的01串连成二进制后按从小到大顺序得到的第i种合并方案得到的新字符,wi表示对应的第i种方案对应获得的分数。 1 ≤ n ≤
300,0 ≤ ci ≤ 1,wi ≥ 1, k ≤ 8
输出描述:
输出一个整数表示答案
示例1
输入
复制
3 2
101
1 10
1 10
0 20
1 30
输出
复制
40
题解:
一开始愣是没看懂题意
长度为k的01串连成二进制后按从小到大顺序,这句话我们用样例来讲就是
00,01,10,11这四个从小到大排列
分数分别是输入的10,10,20,30
参考题解
区间dp+状压dp
dp[l][r][s]表示区间[l,r]合并为s的最小代价
经过推到可以得到:当串的长度为 lenlen 时,最后该串的长度为 (len-1) mod (k-1)+1
区间dp问题就是枚举中间的端点,把区间[l,r]拆分成[l,mid]和[mid+1,r]
状态定义:
f[l][r][S<<1]=max(f[l][r][S<<1],f[l][mid][S]+f[mid+1][r][0]);
f[l][r][S<<1|1]=max(f[l][r][S<<1|1],f[l][mid][S]+f[mid+1][r][1]);
区间[mid+1,r]相当于已经合并完了,因为合并完的值就是0或1,直接加到前面区间[l,mid]上,f[l][mid][S]则是看[l,mid]这段区间合并成多少
当区间长度正好为k时,就直接合并,更新数据(用两个临时变量先储存)
g[c[S]]=max(g[c[S]],f[l][r][S]+w[S]);
区间[l,r]合并为c[S]的最大值,然后
f[l][r][0]=g[0];f[l][r][1]=g[1];
g[]就是临时变量
代码:
#include <bits/stdc++.h>
using namespace std;
#define LL long long
#define INF 0x3f3f3f3f
LL f[310][310][(1<<8)+20];
char s[310];
int b[(1<<8)+20];
LL c[(1<<8)+20];
int main()
{
int n,m,k,i,j;
scanf("%d%d",&n,&k);
scanf("%s",s+1);
for(i=0; i<(1<<k); ++i)
scanf("%d%lld",b+i,c+i);
memset(f,-INF,sizeof(f));
LL inf=f[0][0][0];
for(i=1; i<=n; ++i) f[i][i][s[i]-'0']=0;
for(int len=2; len<=n; ++len)
{
for(i=1; i+len-1<=n; ++i)
{
j=i+len-1;
int l=(j-i)%(k-1);
if(!l) l=k-1;
for(int las=j; las>=i; las-=k-1)
{
for(int S=0; S<(1<<l); ++S)
{
if(f[i][las-1][S]==inf) continue;
if(f[las][j][0]!=inf)
f[i][j][S<<1]=max(f[i][j][S<<1],f[i][las-1][S]+f[las][j][0]);
if(f[las][j][1]!=inf);
f[i][j][S<<1|1]=max(f[i][j][S<<1|1],f[i][las-1][S]+f[las][j][1]);
}
}
if(l==k-1)
{
LL g[2]= {
inf,inf};
for(int S=0; S<(1<<k); ++S)
{
if(f[i][j][S]!=inf)
{
g[b[S]]=max(g[b[S]],f[i][j][S]+c[S]);
}
}
f[i][j][0]=g[0];
f[i][j][1]=g[1];
}
}
}
LL ans=0;
int l=n%(k-1)?n%(k-1):k-1;
for(i=0; i<(1<<l); ++i)
ans=max(ans,f[1][n][i]);
cout<<ans<<endl;
return 0;
}