题目描述

  $pure$在玩一个战略类游戏。现在有一个士兵方阵,每行有若干士兵,每个士兵属于某个兵种。行的顺序不可改变,且每一行中士兵的顺序也不可改变。但由于每一行都有$C$个位置($C$不小于任一行的士兵数),她能够安排每行的士兵依次站在某几个位置上。
  对于每一个士兵,令其前后左右相邻四个位置上有$v$个和他种类相同的士兵,则$pure$会获得$v$的布阵分数。现在$pure$想知道她最多能够获得多少布阵分数。


输入格式

第一行包含两个整数$R,C$,分别表示行数,以及每一行的位置数。
接下来$R$行,每行一个由大写字母构成的字符串,同一字母的士兵为同一种类。


输出格式

一行一个整数,表示$pure$能够获得的最高布阵分数。


样例

样例输入:

2 5
ABBCD
AC

样例输出:

6


数据范围与提示

样例解释:

布阵如下:

ABBCD
A__C_

共获得$6$分。

数据范围:

对于$20\%$的数据,$R\leqslant 3,C\leqslant 4$;
对于$40\%$的数据,$R\leqslant 16$;
对于$100\%$的数据,$R\leqslant 128,C\leqslant 16$,字符串长度不超过$C$。


题解

我们可以只考虑左边和上边的格子,因为兵种一样是相互的,所以最后再乘$2$即可。

先考虑暴力的状压,无非就是枚举上一行的状态,然后再枚举本行的状态,取$\max$,时间复杂度是$\Theta((C_C^{\frac{C}{2}})^2\times C\times R)$的。

然后,我们发现,瓶颈就在于枚举所有的状态,所以我们可以利用轮廓线。

如果你打过插头$DP$,这将非常好理解,枚举行变成了枚举单个格,能对其作出贡献的只有其左边和上边的格了。

代码实现稍繁琐……

时间复杂度:$\Theta(2^C\times C\times R)$。

期望得分:$100$分。

实际得分:$100$分。


代码时刻

#include<bits/stdc++.h>
using namespace std;
int r,c;
char ch[20];
int Map[150][20];
int Mapl[100000][20],Mapr[100000][20];
int dp[2][100000];
bool now;
int ans;
int main()
{
scanf("%d%d",&r,&c);
for(int i=1;i<=r;i++)
{
scanf("%s",ch+1);
Map[i][0]=strlen(ch+1);
for(int j=1;j<=Map[i][0];j++)
Map[i][j]=ch[j]-'A'+1;
}
memset(dp,-0x3f,sizeof(dp));
dp[0][0]=0;
for(int i=0;i<(1<<c);i++)
{
for(int j=2;j<=c+1;j++)
Mapl[i][j]=Mapl[i][j-1]+(i&(1<<(j-2))?1:0);
for(int j=c-1;j;j--)
Mapr[i][j]=Mapr[i][j+1]+(i&(1<<j)?1:0);
}
for(int i=1;i<=r;i++)
{
for(int j=1;j<=c;j++)
{
now^=1;
memset(dp[now],-0x3f,sizeof(dp[now]));
for(int k=0;k<(1<<c);k++)
{
if(Map[i][Mapl[k][j]+1])
dp[now][((((k>>j)<<1)|1)<<(j-1))|(k&((1<<(j-1))-1))]=max(dp[now][((((k>>j)<<1)|1)<<(j-1))|(k&((1<<(j-1))-1))],dp[!now][k]+(((k&(1<<j-2))?Map[i][Mapl[k][j]]:0)==Map[i][Mapl[k][j]+1])+(((k&(1<<j-1))?Map[i-1][Map[i-1][0]-Mapr[k][j]]:0)==Map[i][Mapl[k][j]+1]));
dp[now][((((k>>j)<<1)|0)<<(j-1))|(k&((1<<(j-1))-1))]=max(dp[now][((((k>>j)<<1)|0)<<(j-1))|(k&((1<<(j-1))-1))],dp[!now][k]);
}
}
for(int j=0;j<(1<<c);j++)
if(Mapl[j][c+1]!=Map[i][0])dp[now][j]=-0x3f3f3f3f;
}
for(int i=0;i<(1<<c);i++)
ans=max(ans,dp[now][i]);
printf("%d",(ans<<1));
return 0;
}

rp++

[杂题]:group(状压DP+轮廓线)的更多相关文章

  1. [luoguP3694] 邦邦的大合唱站队/签到题(状压DP)

    传送门 来自kkk的题解: 70分做法:枚举每个学校顺序,暴力. 100分:状压dp.从队列头到尾DP, 状态:f[i]表示i状态下最小的出列(不一致)的个数. 比如f[1101]表示从头到位为1/3 ...

  2. group:状压dp,轮廓线

    神仙题.但是难得的傻孩子cbx没有喊题解,所以也就难得的自己想出来了一个如此神仙的题. 如果是自己想的,说它神仙是不是有点不合适啊..? 反正的确不好像.关键就在于这个标签.颓完标签就差不多会了. % ...

  3. 【专业找水题】状压dp最水题,没有之一

    题目链接 现在代码能力没上升,倒是越来越会找水题了(比例题还水的裸题你值得拥有) 这网站不是针对竞赛的,所以时空限制都很宽松 然后就让我水过去了 对于每个点,包括自己的前m个元素是否取都是一种状态,所 ...

  4. group 状压dp

    应某些人要求,我把标签删掉了 这是一道好题. 一看$c<=16$果断状压,但是怎么压? 一个很显然的思路是,枚举上下两层的状态,每一层的状态极限有$C(c,c/2)$,c=16的时候有13000 ...

  5. nefu1109 游戏争霸赛(状压dp)

    题目链接:http://acm.nefu.edu.cn/JudgeOnline/problemShow.php?problem_id=1109 //我们校赛的一个题,状压dp,还在的人用1表示,被淘汰 ...

  6. 刷题向》关于第一篇状压DP BZOJ1087 (EASY+)

    这是本蒟蒻做的第一篇状压DP,有纪念意义. 这道题题目对状压DP十分友善,算是一道模板题. 分析题目,我们发现可以用0和1代表每一个格子的国王情况, 题目所说国王不能相邻放置,那么首先对于每一行是否合 ...

  7. 【bzoj1087】【互不侵犯King】状压dp裸题(浅尝ACM-D)

    [pixiv] https://www.pixiv.net/member_illust.php?mode=medium&illust_id=54329606 向大(hei)佬(e)势力学(di ...

  8. QDUOJ 来自xjy的签到题(bfs+状压dp)

    来自xjy的签到题   Description 爱丽丝冒险来到了红皇后一个n*n大小的花园,每个格子由'.'或'#'表示,'.'表示爱丽丝可以到达这个格子,‘#’表示爱丽丝不能到达这个格子,爱丽丝每1 ...

  9. POJ 3254 Corn Fields (状压DP,轮廓线DP)

    题意: 有一个n*m的矩阵(0<n,m<=12),有部分的格子可种草,有部分不可种,问有多少种不同的种草方案(完全不种也可以算1种,对答案取模后输出)? 思路: 明显的状压DP啦,只是怎样 ...

随机推荐

  1. mysql定时备份shell脚本

    #!/bin/bash #每天早上4点, mysql备份数据 # backup.sh #crontab -e # * * * /home/erya/run/moniter/mysql_backup.s ...

  2. curl 中关于 CURLINFO_HEADER_SIZE 的 BUG 定位及修复

    curl 官方下载页面 CentOS7 默认安装的 curl 版本太低了,需要升级为最新版. 1. 问题描述 对接了一个接口,用来下载 PDF 文件.使用 curl 下载后,文件老是报错无法打开.接口 ...

  3. curl 和 wget 命令

    1. curl curl 支持 HTTP.HTTPS.FTP 等协议,还支持 POST.cookies.认证.从指定偏移处下载部分文件.User-Agent.限速.文件大小.进度条等特征. 1.1 选 ...

  4. Gradient Descent with Momentum and Nesterov Momentum

    在Batch Gradient Descent及Mini-batch Gradient Descent, Stochastic Gradient Descent(SGD)算法中,每一步优化相对于之前的 ...

  5. tensorflow 之Dataset数据集之批量数据

    ###生成批次数据 import tensorflow as tf '''reapt()生成重复数据集 batch()将数据集按批次组合''' file_name = ['img1','img2',' ...

  6. 初识内存分配ByteBuf

    初识ByteBuf: ByteBuf 是Netty 整个结构里面最为底层的模块,主要负责把数据从底层IO 里面读到ByteBuf,然后传递给应用程序,应用程序处理完成之后再把数据封装成ByteBuf ...

  7. Java-多线程第三篇3种创建的线程方式、线程的生命周期、线程控制、线程同步、线程通信

    1.Java使用Thread类代表线程.     所有的线程对象必须是Thread类或其子类的实例. 当线程继承Thread类时,直接使用this即可获取当前线程,Thread对象的getName() ...

  8. Linux mysql 乱码

    http://www.pc6.com/infoview/Article_63586.html http://itindex.net/detail/41748-linux-mysql-5.5 http: ...

  9. python学习第十二天列表的循环,排序,统计操作方法

    python列表最重要的列表的循环,任何有序列表离不开循环,列表的循环 for  in  range等关键词,还有列表排序,正序,倒序,还有列表每个元素的最大,最小,统计元素的个数等. 1,列表的循环 ...

  10. redhat6.5单用户重置root密码

    (1),按 “e” 键进入该界面,继续按 “e” 键进入下一个界面. (2).上下键选中第二个kernel选项,继续按 “e” 键进行编辑. (3).在新的界面里面加一个空格,再输入“1”:或者输入“ ...