UVa 10618 Fun Game

题目:

http://acm.hust.edu.cn/vjudge/problem/viewProblem.action?id=36035

思路:

  一圈人围坐,给出小球的传递序列,求解最少有多少个人。

问题简单化:如果有一排人,单向传递,给出序列求解最少多少人。那么问题就是:求解一个最短序列使得给出的所有序列都是该序列的连续子序列。

再分别解决以下问题:

  1. 第一个人可以向左向右传递: 每个传递序列有两种情况,分别是正序与逆序。因为不清楚当前序列是向左传还是向右传。
  2. 传递是一个环:规定将第一个序列正向串作为首序列,累计答案的时候只需要算上重叠部分即可。
  3. 输入可能是绕过好几圈的情况:如果绕过好几圈,那么其他的字串因已经被包含而舍弃,dp过程不会进行,最后累计ans时ans=len[0]-重叠(s[0][0],s[0][0])即最小循环节的长度。
  4. 输入保证n>=2:当ans<=1是修改为2

代码:

 

 #include<iostream>
#include<cstring>
#include<vector>
#include<algorithm>
#define FOR(a,b,c) for(int a=(b);a<(c);a++)
using namespace std; const int maxn = ;
const int maxlen= + ; struct Node{
string s,rev;
bool operator <(const Node& rhs) const{
return s.size()<rhs.s.size(); //size由小到大
}
}; int n;
int d[<<maxn][maxn][];
int overlap[maxn][maxn][][];
string s[maxn][];
int len[maxn]; //return a右 && b左的重叠部分
int make_overlap(const string& a,const string& b){
int n1=a.size();
int n2=b.size();
FOR(i,,n1) { //找到最小重合点 即返回最大重合长度
if(n2 + i <= n1) continue;
bool ok=true;
for(int j=;i+j<n1;j++)
if(a[i+j] != b[j]) { ok=false; break; }
if(ok) return n1-i;
}
return ; //无重合
} void init() {
Node nodes[maxn];
FOR(i,,n) {
cin>>nodes[i].s;
nodes[i].rev=nodes[i].s;
reverse(nodes[i].rev.begin(),nodes[i].rev.end()); //反转串
} sort(nodes,nodes+n); //sort
int nn=;
FOR(i,,n) {
bool need=true;
FOR(j,i+,n)
if(nodes[j].s.find(nodes[i].s) != string::npos ||
nodes[j].rev.find(nodes[i].s) != string::npos){ //如果被包含则舍弃
need=false; break;
}
if(need) {
s[nn][]=nodes[i].s;
s[nn][]=nodes[i].rev;
len[nn]=s[nn][].size();
nn++;
}
}
n=nn; //构造重叠数组
FOR(i,,n) FOR(x,,)
FOR(j,,n) FOR(y,,)
overlap[i][j][x][y]=make_overlap(s[i][x],s[j][y]);
} inline void update(int& x,int v) {
if(x< || v<x) x=v;
} void solve() {
memset(d,-,sizeof(d));
d[][][]=len[]; //始终把s[0][0]放在首位 int full=<<n;
FOR(s,,full)
FOR(i,,n) FOR(x,,) if(d[s][i][x]>=) //已求过集合中的枚举
FOR(j,,n) if(!((<<j) & s)) FOR(y,,) //刷表更新的目标枚举
update(d[s|(<<j)][j][y], d[s][i][x] + len[j]-overlap[i][j][x][y]); int ans=-; full--;
FOR(i,,n) FOR(x,,) {
if(d[full][i][x]<) continue;
update(ans, d[full][i][x] - overlap[i][][x][]); //ans累计 枚举尾字串并减去与第一个串的重叠部分
}
if(ans<=) cout<<"2\n"; //题目中明确给出 n>=2
else
cout<<ans<<"\n";
}
int main(){
while(cin>>n && n) {
init();
solve();
}
return ;
}

【暑假】[深入动态规划]UVa 10618 Fun Game的更多相关文章

  1. 【暑假】[深入动态规划]UVa 10618 Fixing the Great Wall

    UVa 10618 Fixing the Great Wall 题目:  http://acm.hust.edu.cn/vjudge/problem/viewProblem.action?id=361 ...

  2. 【暑假】[深入动态规划]UVa 10618 Tango Tango Insurrection

    UVa 10618 Tango Tango Insurrection 题目: Problem A: Tango Tango Insurrection You are attempting to lea ...

  3. 【暑假】[深入动态规划]UVa 10618 The Bookcase

    UVa 12099  The Bookcase 题目: http://acm.hust.edu.cn/vjudge/problem/viewProblem.action?id=42067 思路:    ...

  4. 【暑假】[深入动态规划]UVa 1628 Pizza Delivery

    UVa 1628 Pizza Delivery 题目: http://acm.hust.edu.cn/vjudge/problem/viewProblem.action?id=51189 思路:    ...

  5. 【暑假】[深入动态规划]UVa 1380 A Scheduling Problem

     UVa 1380 A Scheduling Problem 题目: http://acm.hust.edu.cn/vjudge/problem/viewProblem.action?id=41557 ...

  6. 【暑假】[深入动态规划]UVa 12170 Easy Climb

    UVa 12170 Easy Climb 题目: http://acm.hust.edu.cn/vjudge/problem/viewProblem.action?id=24844 思路:  引别人一 ...

  7. 【暑假】[深入动态规划]UVa 1627 Team them up!

    UVa 1627 Team them up! 题目: Team them up! Time Limit: 3000MS   Memory Limit: Unknown   64bit IO Forma ...

  8. 【暑假】[深入动态规划]UVa 1412 Fund Management

    UVa 1412 Fund Management 题目: UVA - 1412 Fund Management Time Limit: 3000MS   Memory Limit: Unknown   ...

  9. ACM - 动态规划 - UVA 1347 Tour

    UVA 1347 Tour 题解 题目大意:有 \(n\) 个点,给出点的 \(x\).\(y\) 坐标.找出一条经过所有点一次的回路,从最左边的点出发,严格向右走,到达最右点再严格向左,回到最左点. ...

随机推荐

  1. 1027: [JSOI2007]合金 - BZOJ

    Description 某公司加工一种由铁.铝.锡组成的合金.他们的工作很简单.首先进口一些铁铝锡合金原材料,不同种类的原材料中铁铝锡的比重不同.然后,将每种原材料取出一定量,经过融解.混合,得到新的 ...

  2. 如何使用 XSD

    如何使用 XSD 一个简单的 XML 文档: 请看这个名为 "note.xml" 的 XML 文档: <?xml version="1.0"?> & ...

  3. 优化SQL Server数据库查询方法

    SQL Server数据库查询速度慢的原因有很多,常见的有以下几种: 1.没有索引或者没有用到索引(这是查询慢最常见的问题,是程序设计的缺陷) 2.I/O吞吐量小,形成了瓶颈效应. 3.没有创建计算列 ...

  4. Discuz云平台站点信息同步失败,An unknown error occurred. May be DNS Error.

    站点信息同步失败 An unknown error occurred. May be DNS Error. (ERRCODE:1) 经过Discuz教程网(http://www.1314study.c ...

  5. hdu 4559 涂色游戏 博弈论

    构造SG函数:sg[i]表示2*i的sg值!! 代码如下: #include<iostream> #include<stdio.h> #include<algorithm ...

  6. 1.Getting Started with ASP.NET MVC 5

    Getting Started Start by installing and running Visual Studio Express 2013 for Web or Visual Studio ...

  7. 多线程(一)NSThread

    iOS中多线程的实现方案: 技术 语言 线程生命周期 使用频率 pthread C 程序员自行管理 几乎不用 NSthread OC 程序员自行管理 偶尔使用 GCD C 自动管理 经常使用 NSOp ...

  8. 图模型的统计推断 inference in graphical models(马尔科夫链的推断)

    有关因子图(factor graphs)以及其在sum product 算法,max-algorithm中的应用,将在一下篇博客中分享. 谢谢您的关注,欢迎提出意见问题.

  9. uva 11817 - Tunnelling the Earth

    题意:从地球上的一个点到另一个点,求两点的球面距离和直线距离之差.假定地球是正球体,半径为6371009米. #include<iostream> #include<cmath> ...

  10. 基于Android Studio搭建Android应用开发环境

    备注:电脑是windows xp系统 1.     安装JDK和环境变量设置 JDK是java development kit,Java JDK下载地址 http://www.oracle.com/t ...