Fun Game

https://odzkskevi.qnssl.com/8d698323a1e07d605cdeea708ee8a01d?v=1508703139

【题解】

不难发现如果一个串的原串或反转串是另一个串的子串,那么这个串是没有用的

我们把他剔除出去

如果此时只有一个串,暴力枚举解检查即可(网上很多写法是KMP。。不是很明白,没仔细看他们代码

多个串则状压DP

dp[s][i][0/1]表示s串已经选了,最后一个串是i,i是正着/倒着的,最大重叠字母数

刷表法转移即可

如何处理圈?我们强行在第一个串的地方断开,按照第一个串的正着的方向为圈的传递方向即可,

最后的时候枚举最后一个串跟第一个串交一下算答案

 #include <iostream>
#include <algorithm>
#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <vector>
#define min(a, b) ((a) < (b) ? (a) : (b))
#define max(a, b) ((a) > (b) ? (a) : (b)) inline void swap(int &a, int &b)
{
long long tmp = a;a = b;b = tmp;
} inline void read(int& x)
{
x = ;char ch = getchar(), c = ch;
while(ch < '' || ch > '')c = ch, ch = getchar();
while(ch <= '' && ch >= '')x = x * + ch - '', ch = getchar();
} const int INF = 0x3f3f3f3f;
const int MAXN = ;
const int MAXNUM = ; int dp[( << MAXN) + ][MAXN + ][], die[MAXN][][MAXN][], n, ma, sum, len[MAXN], cnt[MAXN], tot, b[MAXN];
char s[MAXN][MAXNUM]; inline void init()
{
memset(dp, , sizeof(dp));
memset(die, , sizeof(die));
tot = ;
sum = ;
memset(b, , sizeof(b));
memset(len, , sizeof(len));
memset(cnt, , sizeof(cnt));
//check j是否包含在i中
for(register int i = ;i <= n;++i)
{
scanf("%s", s[i] + );
len[i] = strlen(s[i] + );
}
for(register int i = ;i <= n;++ i)
for(register int j = ;j <= n;++ j)
{
if(i == j || b[i] || b[j] || len[j] > len[i])continue;
//正着配
for(register int a = ;a <= len[i] - len[j] + ;++ a)
{
int tmp1 = a;
while(s[i][tmp1] == s[j][tmp1 - a + ] && tmp1 - a + <= len[j]) ++ tmp1;
if(tmp1 - a + > len[j]) b[j] = ;
}
//倒着配
for(register int a = len[i];a >= len[j];-- a)
{
int tmp1 = a, p = ;
while(s[i][tmp1] == s[j][p] && p <= len[j]) -- tmp1, ++ p;
if(p > len[j]) b[j] = ;
}
}
for(register int i = ;i <= n;++ i)
if(!b[i])
cnt[++ tot] = i, sum += len[i];
//j跟在i后面重叠部分大小
for(register int p = ;p <= tot;++ p)
for(register int q = ;q <= tot;++ q)
{
int i = cnt[p], j = cnt[q];
if(i == j)continue;
//0 0
for(register int a = ;a <= len[i];++ a)
{
int b = , tmp = a;
while(s[i][tmp] == s[j][b] && tmp <= len[i] && b <= len[j])++ tmp, ++ b;
if(tmp > len[i])
{
die[p][][q][] = len[i] - a + ;
break;
}
}
//0 1
for(register int a = ;a <= len[i];++ a)
{
int b = len[j], tmp = a;
while(s[i][tmp] == s[j][b] && tmp <= len[i] && b >= )++ tmp, -- b;
if(tmp > len[i])
{
die[p][][q][] = len[i] - a + ;
break;
}
}
//1 0
for(register int a = len[i];a >= ;-- a)
{
int b = , tmp = a;
while(s[i][tmp] == s[j][b] && tmp >= && b <= len[j])-- tmp, ++ b;
if(tmp < )
{
die[p][][q][] = a;
break;
}
}
//1 1
for(register int a = len[i];a >= ;-- a)
{
int b = len[j], tmp = a;
while(s[i][tmp] == s[j][b] && tmp >= && b >= )-- tmp, -- b;
if(tmp < )
{
die[p][][q][] = a;
break;
}
}
}
ma = << tot;
} int main()
{
while(scanf("%d", &n) != EOF && n)
{
init();
int flag = ;
if(tot == )
{
int x = cnt[tot];
for(register int i = ;i <= len[x];++ i)
{
for(register int j = ;j <= len[x];++ j)
{
if(j + i - > len[x]) break;
int p1 = , p2 = j, rank = ;
while(s[x][p1] == s[x][p2] && rank < len[x])
{
++ p1, ++ p2, ++ rank;
if(p1 > len[x]) p1 = ;
if(p2 > j + i - ) p2 = ;
}
if(rank >= len[x])
{
printf("%d\n", max(, i));
flag = ;
break;
}
}
if(flag) break;
}
if(flag) continue;
}
//dp[S][i][0/1]表示以1号字符串为头,已经选了S,最后一个是i的正/反状态的最大折叠数
//dp[S | k][k][p] = max(dp[S | k][k][p], dp[S][i][p'] + die[i][p'][k][p])
for(register int S = ;S < ma;++ S)
for(register int i = ;i <= tot;++ i)/*分别用dp[S][i][0]和dp[S][i][1]去更新*/
{
if(!(S & ))continue;
if(!(S & ( << (i - )))) continue;
for(register int k = ;k <= tot;++ k)
{
if(S & ( << (k - ))) continue;
if(S == )
{
dp[S | ( << (k - ))][k][] = max(dp[S | ( << (k - ))][k][], dp[S][i][] + die[i][][k][]);
dp[S | ( << (k - ))][k][] = max(dp[S | ( << (k - ))][k][], dp[S][i][] + die[i][][k][]);
}
else
{
dp[S | ( << (k - ))][k][] = max(dp[S | ( << (k - ))][k][], max(dp[S][i][] + die[i][][k][], dp[S][i][] + die[i][][k][]));
dp[S | ( << (k - ))][k][] = max(dp[S | ( << (k - ))][k][], max(dp[S][i][] + die[i][][k][], dp[S][i][] + die[i][][k][]));
}
}
}
int ans = ;
for(register int i = ;i <= tot;++i)
ans = max(ans, max(dp[ma - ][i][] + die[i][][][], dp[ma - ][i][] + die[i][][][]));
printf("%d\n", max(, sum - ans));
}
return ;
}

UVA1204

最新文章

  1. RF 基本方法
  2. [dpdk] 熟悉SDK与初步使用 (三)(IP Fragmentation源码分析)
  3. Quartus II USB-Blaster驱动解决
  4. iOS 项目中用到的一些开源库和第三方组件
  5. EXCL poi导入
  6. 天气预报数据API
  7. IIS7显示ASP的详细错误信息到浏览器
  8. Qt之窗口动画(下坠、抖动、透明度)
  9. QTP10&amp;QTP11&amp;UFT11.5的安装和破解
  10. JBoss 系列十一:JBoss Cluster Framework Demo 介绍
  11. Windows Live Write 日志客户端
  12. Net开源网络爬虫
  13. Unity3D中录制和输出wav文件
  14. Chrome F12调试
  15. golang:mime.Decode、mime.DecodeHeader
  16. python 读写TXT,安装pandas模块。
  17. JMeter安装+配置+运行
  18. 普通的jdbc事务在插入数据后 下面的代码报错时 数据不会回滚 但是 spring的事务会回滚
  19. SpringBoot 线程池配置 实现AsyncConfigurer接口方法
  20. Web—09-正则表达式

热门文章

  1. mysql 存中文失败问题
  2. 移动端自定义键盘的vue组件 ----keyboard
  3. hession RMI 远程调用
  4. Error:【SLF4J: Class path contains multiple SLF4J bindings.】
  5. Python爬虫笔记【一】模拟用户访问之设置请求头 (1)
  6. vue 引入css及注意事项
  7. 不同版本springboot端点开启方法
  8. pickle序列化一个函数,将fun()存入文件
  9. CODE[VS]1372:DNA
  10. 原生微信小程序数据渲染