[HAOI2016]字符合并

时间限制: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;
}