Written with StackEdit.

Description

有一个长度为 \(n\) 的 \(01\) 串,你可以每次将相邻的 \(k\) 个字符合并,得到一个新的字符并获得一定分数。得到的新字符和分数由这\(k\) 个字符确定。你需要求出你能获得的最大分数。

Input

第一行两个整数\(n,k\)。接下来一行长度为n的01串,表示初始串。接下来\(2^k\)行,每行一个字符\(c_i\)和一个整数\(w_i\),\(c_i\)表示长度为\(k\)的\(01\)串连成二进制后按从小到大顺序得到的第i种合并方案得到的新字符,\(w_i\)表示对应的第\(i\)种方案对应获得的分数。

Output

输出一个整数表示答案.

Sample Input

3 2

101

1 10

1 10

0 20

1 30

Sample Output

40

  • 第3行到第6行表示长度为2的4种01串合并方案。00->1,得10分,01->1得10分,10->0得20分,11->1得30分。

Hint

\(1\leq n\leq 300,0\leq ci\leq 1,1\leq w_i,k\leq8.\)

Solution

并不会做...于是临摹zxTLEdalao的题解...然后自己似乎懂了一点.

  • 这道题\(k\leq8\),做出一副状压的嘴脸虽然的确也用到了状压,但解决本题的核心在于区间\(dp\).
  • 考虑令\(f[l][r][p]\)表示将原串\([l,r]\)合并为\(t\)能获得的最大收益.
  • 显然是满足无后效性的.合并起来之后就不再关心合并过程了.
  • 枚举中间的端点\(mid\),使得\(mid\)右边的子串合并起来是\(t\)的最后一位数字,左边的子串合并起来是\(t\)前面的数字.
  • 注意到,合并时只能连续的\(k\)个数字合并,那么可以合并成\(1\)位数字的子串长度一定为\(1,k,2k-1......\),枚举\(mid\)时相差为\(k-1\)即可.
  • 特殊情况:从\(l\)到\(r\)恰好有\(k\)个数,这时我们直接合并,用辅助数组\(g\)求出此时的\(f\),
#include<bits/stdc++.h>
#define inf (1LL<<60)//防止相加爆long long
using namespace std;
typedef long long ll;
inline ll read()
{
int out=0,fh=1;
char jp=getchar();
while ((jp>'9'||jp<'0')&&jp!='-')
jp=getchar();
if (jp=='-')
{
fh=-1;
jp=getchar();
}
while (jp>='0'&&jp<='9')
{
out=out*10+jp-'0';
jp=getchar();
}
return out*fh;
}
inline void upd(ll &x,ll y)
{
x=max(x,y);
}
const int MAXN=329;
const int MAXS=(1<<8)+10;
int a[MAXN];
int n,k,lim;
char buf[MAXN];
ll w[MAXS],c[MAXS];
ll f[MAXN][MAXN][MAXS];//f[l][r][k]:将原串的[l,r]合并为t能获得的最大分数
int main()
{
n=read(),k=read();
scanf("%s",buf+1);
for(int i=1;i<=n;++i)
a[i]=(buf[i]=='1');
lim=(1<<k)-1;
for(int i=0;i<=lim;++i)
c[i]=read(),w[i]=read();
ll ans=0;
for(int i=1;i<=n;++i)
for(int j=1;j<=n;++j)
for(int p=0;p<=lim;++p)
f[i][j][p]=-inf;
for(int i=n;i>=1;--i)
for(int j=i;j<=n;++j)
{
if(i==j)
{
f[i][j][a[i]]=0;
continue;
}
int len=j-i;
len%=k-1;
if(!len)
len=k-1;
for(int mid=j;mid>i;mid-=k-1)
for(int op=0;op<(1<<len);++op)
{
upd(f[i][j][op<<1],f[i][mid-1][op]+f[mid][j][0]);
upd(f[i][j][op<<1|1],f[i][mid-1][op]+f[mid][j][1]);
}
if(len==k-1)
{
ll g[2];
g[0]=g[1]=-inf;
for(ll op=0;op<(1<<k);++op)
upd(g[c[op]],f[i][j][op]+w[op]);
f[i][j][0]=g[0];
f[i][j][1]=g[1];
}
}
for(int i=0;i<=lim;++i)
upd(ans,f[1][n][i]);
printf("%lld\n",ans);
return 0;
}

bzoj 4565 字符合并的更多相关文章

  1. 【BZOJ】4565: [Haoi2016]字符合并

    4565: [Haoi2016]字符合并 Time Limit: 20 Sec  Memory Limit: 256 MBSubmit: 690  Solved: 316[Submit][Status ...

  2. 「BZOJ 4565」「HAOI 2016」字符合并「区间状压DP」

    题意 给一个长度为\(n(\leq 300)\)的\(01\)串,每次可以把\(k(\leq 8)\)个相邻字符合并,得到新字符和一定分数,最大化最后的得分 题解 考虑设计dp:\(dp[S][i][ ...

  3. Linux:不同文件相同列字符合并文件(awk函数)

    存在file1.txt,其内容如下: H aa 0 0 1 -9 H bb 0 0 2 -9 H cc 0 0 2 -9 存在file2.txt,其内容如下: H aa 0 0 0 -9 asd qw ...

  4. SQL SERVER 字符合并多行为一列

    [字符合并多行为一列] 思路1:行转列,在与字符拼接(适用每组列数名相同) 思路2:转xml,去掉多余字符(适用所有) 假设兴趣表Hobbys Name Hobby 小张 打篮球 小张 踢足球 Nam ...

  5. 【BZOJ4565】【HAOI2016】字符合并 [状压DP][区间DP]

    字符合并 Time Limit: 20 Sec  Memory Limit: 256 MB[Submit][Status][Discuss] Description 有一个长度为 n 的 01 串,你 ...

  6. BZOJ4565 [Haoi2016]字符合并

    题意 有一个长度为\(n\)的\(01\)串,你可以每次将相邻的\(k\)个字符合并,得到一个新的字符并获得一定分数.得到的新字符和分数由这\(k\)个字符确定.你需要求出你能获得的最大分数. \(n ...

  7. 「HAOI2016」字符合并

    「HAOI2016」字符合并 题意: ​ 有一个长度为\(n\)的\(01\)串,你可以每次将相邻的\(k\)个字符合并,得到一个新的字符并获得一定分数.得到的新字符和分数由这\(k\)个字符确定.你 ...

  8. [Haoi2016]字符合并 题解

    tijie 时间限制: 2 Sec  内存限制: 256 MB 题目描述 有一个长度为 n 的 01 串,你可以每次将相邻的 k 个字符合并,得到一个新的字符并获得一定分数.得到的新字 符和分数由这 ...

  9. 『字符合并 区间dp 状压dp』

    字符合并 Description 有一个长度为 n 的 01 串,你可以每次将相邻的 k 个字符合并,得到一个新的字符并获得一定分数.得到的新字符和分数由这 k 个字符确定.你需要求出你能获得的最大分 ...

随机推荐

  1. Sublime Text 3 快捷键 一览

    Sublime Text 3 快捷键精华版 Ctrl+Shift+P:打开命令面板 Ctrl+P:搜索项目中的文件 Ctrl+G:跳转到第几行 Ctrl+W:关闭当前打开文件 Ctrl+Shift+W ...

  2. 请求库之requests,selenium

    requests模块 一.介绍 #介绍:使用requests可以模拟浏览器的请求,比起之前用到的urllib,requests模块的api更加便捷(本质就是封装了urllib3) #注意:reques ...

  3. Metasploit – 内网连接

    0x00 问题描述 在渗透测试时,metasploit往往作为后渗透工具,(因为远程溢出越来越少).我一般都是在获得一个webshell后,来使用metasploit进行信息采集,或者内网扫描等操作. ...

  4. 谈谈对Canal(增量数据订阅与消费)的理解

    概述 canal是阿里巴巴旗下的一款开源项目,纯Java开发.基于数据库增量日志解析,提供增量数据订阅&消费,目前主要支持了mysql(也支持mariaDB). 起源:早期,阿里巴巴B2B公司 ...

  5. Linux下MySQL 5.6.24的编译安装与部署

    MySQL 5.6正式版发布了,相对于5.5版本作出了不少改进,其源码安装配置方式也有所变化,本文根据实际操作,不断尝试,精确还原了安装的具体步骤. 在Linux下安装MySQL前,先确认卸载系统自带 ...

  6. mysql分库分表(一)

    mysql分库分表 参考: https://blog.csdn.net/xlgen157387/article/details/53976153 https://blog.csdn.net/cleve ...

  7. BZOJ4487 [Jsoi2015]染色问题

    BZOJ4487 [Jsoi2015]染色问题 题目描述 传送门 题目分析 发现三个限制,大力容斥推出式子是\(\sum_{i=0}^{N}\sum_{j=0}^{M}\sum_{k=0}^{C}(- ...

  8. 解决org.apache.hadoop.io.nativeio.NativeIO$Windows.access0(Ljava/lang/String;I)Z

    这个问题来的有点莫名奇妙,之前我的hadoop运行一直是正常的,某一天开始运行Mapreduce就报这个错. 试过很多种方法都没有用,比如 1.path环境变量2.Hadoop bin目录下hadoo ...

  9. Win7.还原默认打开方式

    1.win7还原默认打开方式_百度知道.html(https://zhidao.baidu.com/question/1668708948433912307.html) Windows7:[47]打开 ...

  10. java.util.logging.Logger_01

    1.参考网址 1.1.java.util.logging.Logger使用详解 http://lavasoft.blog.51cto.com/62575/184492 1.2.Java内置Logger ...