题意

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

\(n \leq 300,k \leq 8,1 \leq w_i \leq 10^9\)

分析

参照xyz32768的题解。

区间合并让人想到区间 dp ,而 \(k≤8\) 又让人想到状压 dp 。

我们考虑合二为一。

\(f[l][r][S]\) 表示将区间 \([l,r]\) 内的字符不断合并,最后变成串 \(S\) 的最大收益。

( \(S\) 是一个长度为 \((r−l)mod(k−1)+1\) 的 \(01\) 串)

(由于每次合并会减少 \(k−1\) 个字符,故 \(S\) 的长度固定)

考虑 \(S\) 的每个字符,它们都是由原串的一个区间逐渐压缩成的。

故 \(S\) 的每个字符互相独立,互不影响。

我们就枚举一个 \(mid∈[l,r)\) ,表示 \(S\) 的最后一个字符是由原串的区间 \((mid,r]\) 压缩成的。

这时候就有一个非常传统的区间 dp 转移了!

以下把 \(mg(S,x)\) 定义为 \((S<<1)|x\) ,即在 \(S\) 的后面插入 \(x\) 。 \(x∈{0,1}\) 。

\[f[l][r][mg(S,x)]=max(f[l][r][mg(S,x)],f[l][mid][S]+f[mid+1][r][x])
\]

其中 \(x∈{0,1}\) 。

注意上面针对的是 \(|S|=(r−l)mod(k−1)+1<k−1\) 的情况。

如果 \(|S|=k−1\) ,那么 \([l,mid]\) 会和 \((mid,r]\) 组成一个长度为 \(k\) 的串,还可以再次合并。

故当 \(|S|=k−1\) 时:

\[f[l][r][c[mg(S,x)]]=max(f[l][r][c[mg(S,x)]],f[l][mid][S]+f[mid+1][r][x]+w[mg(S,x)])
\]

同样 \(x∈{0,1}\)

理论复杂度\(O(2^k \cdot n^3)\) ,但实际状态没有那么多。

代码

#include<iostream>
#include<cstdio>
#include<cstdlib>
#include<cmath>
#include<set>
#include<map>
#include<queue>
#include<stack>
#include<algorithm>
#include<cstring>
#define rg register
#define il inline
#define co const
template<class T>T read()
{
T data=0;
int w=1;
char ch=getchar();
while(!isdigit(ch))
{
if(ch=='-')
w=-1;
ch=getchar();
}
while(isdigit(ch))
{
data=data*10+ch-'0';
ch=getchar();
}
return data*w;
}
template<class T>T read(T&x)
{
return x=read<T>();
}
using namespace std;
typedef long long ll; co int MAXN=300,MAXK=8;
int n,k;
int a[MAXN];
int c[1<<MAXK],w[1<<MAXK];
ll f[MAXN][MAXN][1<<MAXK]; int main()
{
// freopen(".in","r",stdin);
// freopen(".out","w",stdout);
read(n);read(k);
for(int i=0;i<n;++i)
read(a[i]);
for(int i=0;i<(1<<k);++i)
{
read(c[i]);read(w[i]);
} memset(f,-1,sizeof f);
for(int i=0;i<n;++i)
f[i][i][a[i]] = 0;
for(int i=n-1;i>=0;--i)
for(int j=i+1;j<n;++j)
{
int len = (j - i) % (k - 1) + 1;
if(len > 1)
{
for(int mid = i + len - 2;mid <= j - 1;mid += k - 1)
for(int s = 0;s < (1 << (len - 1));++s)
{
if(f[i][mid][s]==-1)
continue;
if(f[mid+1][j][0]!=-1)
f[i][j][s<<1] = max(f[i][j][s<<1],f[i][mid][s]+f[mid+1][j][0]);
if(f[mid+1][j][1]!=-1)
f[i][j][s<<1|1] = max(f[i][j][s<<1|1],f[i][mid][s]+f[mid+1][j][1]);
}
}
else
{
for(int s = 0;s < (1 << (k - 1));++s)
for(int mid = i + k - 2;mid <= j - 1;mid += k - 1)
{
if(f[i][mid][s]==-1)
continue;
if(f[mid+1][j][0]!=-1)
f[i][j][c[s<<1]]=max(f[i][j][c[s<<1]],f[i][mid][s]+f[mid+1][j][0]+w[s<<1]);
if(f[mid+1][j][1]!=-1)
f[i][j][c[s<<1|1]]=max(f[i][j][c[s<<1|1]],f[i][mid][s]+f[mid+1][j][1]+w[s<<1|1]);
}
}
}
ll ans=-1;
for(int i=0;i<(1<<k);++i)
{
// cerr<<i<<" f="<<f[0][n-1][i]<<endl;
ans=max(ans,f[0][n-1][i]);
}
printf("%lld\n",ans);
return 0;
}

BZOJ4565 [Haoi2016]字符合并的更多相关文章

  1. BZOJ4565 HAOI2016字符合并(区间dp+状压dp)

    设f[i][j][k]为将i~j的字符最终合并成k的答案.转移时只考虑最后一个字符是由哪段后缀合成的.如果最后合成为一个字符特殊转移一下. 复杂度看起来是O(n32k),实际常数极小达到O(玄学). ...

  2. [BZOJ4565][HAOI2016]字符合并(区间状压DP)

    https://blog.csdn.net/xyz32768/article/details/81591955 首先区间DP和状压DP是比较明显的,设f[L][R][S]为将[L,R]这一段独立操作最 ...

  3. 2018.10.25 bzoj4565: [Haoi2016]字符合并(区间dp+状压)

    传送门 当看到那个k≤8k\le 8k≤8的时候就知道需要状压了. 状态定义:f[i][j][k]f[i][j][k]f[i][j][k]表示区间[i,j][i,j][i,j]处理完之后的状态为kkk ...

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

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

  5. 【BZOJ4565】 [Haoi2016]字符合并

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

  6. [Haoi2016]字符合并 题解

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

  7. 题解 [HAOI2016]字符合并

    题目传送门 Description 有一个长度为 \(n\) 的 \(01\) 串,你可以每次将相邻的 k 个字符合并,得到一个新的字符并获得一定分数. 得到的新字符和分数由这 k 个字符确定.你需要 ...

  8. 【BZOJ 4565】 [Haoi2016]字符合并 区间dp+状压

    考试的时候由于总是搞这道题导致爆零~~~~~(神™倒序难度.....) 考试的时候想着想着想用状压,但是觉得不行又想用区间dp,然而正解是状压着搞区间,这充分说明了一件事,状压不是只是一种dp而是一种 ...

  9. [HAOI2016]字符合并

    Luogu3736 很容易想到直接DP,关键是枚举顺序. \(1.\)设后一段构成最后一个点,前一段构成前面的点,那么能得到\(1\)个点的数量要求 : \(1,k,2k-1...\)相差\(k-1\ ...

随机推荐

  1. Memory Manager surface area changes in SQL Server 2012

    here were various changes to memory related DMVs, DBCC memory status, and Perfmon counters in SQL Se ...

  2. wireshark抓取mysql数据包

    最近在学习搭建数据库服务,因为跟产品相关所以需要从流量中拿到mysql的数据包.然后就想着在本机搭建mysql数据库,然后连接,用wireshark抓就行了. MySQL搭建用的是XAMPP,想说XA ...

  3. HDU - 5988The 2016 ACM-ICPC Asia Qingdao Regional ContestG - Coding Contest 最小费用流

    很巧妙的建边方式 题意:有n个区域,每个区域有一些人数si和食物bi,区域之间有m条定向路径,每条路径有人数通过上限ci.路径之间铺了电线,每当有人通过路径时有pi的概率会触碰到电线,但是第一个通过的 ...

  4. 不使用构造方法创建Java对象: objenesis的基本使用方法

    转载:http://blog.csdn.net/codershamo/article/details/52015206 objenesis简介: objenesis是一个小型Java类库用来实例化一个 ...

  5. day5-os、sys模块

    一.概述 开发运维相关支撑系统现今已成为Devops下的一大热门领域,Python在这方面也有着自己独到的优势.这类场景以及其他一些场景下,需要调用一些操作系统的接口,这就涉及到今天要讲述的OS模块和 ...

  6. WAF的实现

    文章来源:http://danqingdani.blog.163.com/blog/static/1860941952014101723845500/ 本篇文章从WAF产品研发的角度来YY如何实现一款 ...

  7. APUE学习笔记——10信号——信号接口函数 signal 和 sigaction

    signal函数     signal函数是早起Unix系统的信号接口,早期系统中提供不可靠的信号机制.在后来的分支中,部分系统使用原来的不可靠机制定义signal函数,如 Solaris 10 .而 ...

  8. 利用xcopy在复制文件或文件夹的时候保留其权限

    当用 Windows Explorer 复制或移动文件和文件夹时,文件或文件夹上设置的权限可能会发生改变.例如,当在一个 NTFS文件系统卷内或在两个 NTFS 卷之间复制一个文件时,Windows将 ...

  9. 《利用Python进行数据分析》笔记---第2章--来自bit.ly的1.usa.gov数据

    写在前面的话: 实例中的所有数据都是在GitHub上下载的,打包下载即可. 地址是:http://github.com/pydata/pydata-book 还有一定要说明的: 我使用的是Python ...

  10. 解决在django中应用keras模型时出现的ValueError("Tensor %s is not an element of this graph." % obj)问题

    用keras训练好模型,再在django初始化加载模型,这个过程没有问题,但是在调用到模型执行model.predict()的时候就报错: raise ValueError("Tensor ...