题目描述

  $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. OJ测试数据追溯方法

    https://blog.csdn.net/iwts_24/article/details/79240987 我是从这个博客知道还有从比赛官方网站扒测试数据的方法,但是下面的链接有些是失效的. 所以我 ...

  2. jQury+Ajax与C#后台交换数据

    -------------------------------------------jQury+Ajax调用后台方法----------------------------------------- ...

  3. ELK日志分析系统之elasticsearch7.x最新版安装与配置

    1.Elasticsearch 1.1.elasticsearch的简介 ElasticSearch是一个基于Lucene的搜索服务器.它提供了一个分布式多用户能力的全文搜索引擎,基于RESTful ...

  4. #3831 TJOI2013单词

    WOJ#3831 TJOI2013单词 题面 某人读论文,一篇论文是由许多单词组成.但他发现一个单词会在论文中出现很多次,现在想知道每个单词分别在论文中出现多少次. 输入 第一个一个整数 N ,表示有 ...

  5. python学习第四天基本数据类型 int,string,bool

    python跟其他编程语言一样,拥有基本的数据类型,计算机 只能识别0101,python是解释语言,有其他的解释器 python整型 int a=10 type(a) "int| pyth ...

  6. iBatis——自动生成DAO层接口提供操作函数(详解)

    iBatis——自动生成DAO层接口提供操作函数(详解) 在使用iBatis进行持久层管理时,发现在使用DAO层的updateByPrimaryKey.updateByPrimaryKeySelect ...

  7. 09-python的面向对象

    # 1. 面向对象概述(ObjectOriented,OO) - OOP思想 - 接触到任意一个任务,首先想到的是任务这个世界的构成,是由模型构成的 - 几个名词 - OO:面向对象 - OOA:面向 ...

  8. 【学习总结】gcc和gdb

    目录 <> vim.gcc.gdb: gcc: gcc和g++是c/c++的linux系统集成的编译器,源文件的后缀应为 .C/.cpp/.c++/.cc等 编译器可以将C.C++等语言源 ...

  9. k3 cloud中出现合计和汇总以后没有显示出来,合价要新增一行以后才出现值

    解决办法:找到对应字段,把及时触发值更新事件打上勾

  10. Java加密与解密的艺术 读书心得

    现在项目中加密与解密的方式很多,很早就想整理一下Java中加密与解密的方式,读完<<Java加密与解密的艺术>>一书.借此机会梳理一下这方面的知识点 一.基础普及 安全技术目标 ...