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

UVA1204 Fun Game的更多相关文章

随机推荐

  1. 前端面试题之一JAVASCRIPT(理论类)

    一.请描述一下 cookies.sessionstorage .localstorage 和session的区别?(1)cookie是网站为了标示用户身份而储存在用户本地终端(client side) ...

  2. opencv-Mat数据类型及位数总结

    转自:http://blog.sina.com.cn/s/blog_662c7859010105za.html 在OpenCV里面,许多数据结构为了达到內存使用的最优化,通常都会用它最小上限的空间来分 ...

  3. Cesium官方教程11--建模人员必读

    原文地址:https://cesium.com/blog/2014/12/15/gltf-tips-for-artists/ 这篇文章是Branden Coker, an artist from AG ...

  4. GitHub for Visual Studio使用讲解

    从VS2015起(应该是吧?),微软已经在VS中集成了GitHub,方便开发者对项目进行版本控制. 扩展包下载地址:https://aka.ms/ghfvs 其实VS2015的安装包中已经自带了这个扩 ...

  5. Java基础(spring事物和锁)

    使用步骤: 步骤一.在spring配置文件中引入<tx:>命名空间<beans xmlns="http://www.springframework.org/schema/b ...

  6. Leetcode 345 Reverse Vowels in a String

    两个for 第一个for将每一个元音依次存放进一个char数组 第二个for,每检测到元音,就从char数尾部开始,依次赋值 如何检测元音呢?当然写一个冗长的if(),不过我们有更好的选择 hashs ...

  7. Linux自动化工具之crontab (windows需要手动配置相关服务,具体百度)

    //有的shell解释器是/bin/tcsh而不是传统的/bin/bash.而且两者语法有些差异,注意避免.比如设置变量tcsh是set 变量 =`****`   1.crontab是什么,是linu ...

  8. 如何正确使用 Flink Connector?

    本文主要分享 Flink connector 相关内容,分为以下三个部分的内容:第一部分会首先介绍一下 Flink Connector 有哪些.第二部分会重点介绍在生产环境中经常使用的 kafka c ...

  9. PHP--时间搜索插件封装

    /** * 时间搜索插件封装 * anthor qinpeizhou * @param $timeset 时间格式 * @param $time sql语句中所需要搜索的time字段名称 * @par ...

  10. light oj 1079 01背包

    #include <iostream> #include <algorithm> #include <cstring> #include <cstdio> ...