【暑假】[深入动态规划]UVa 10618 Fun Game
UVa 10618 Fun Game
题目:
http://acm.hust.edu.cn/vjudge/problem/viewProblem.action?id=36035
思路:
一圈人围坐,给出小球的传递序列,求解最少有多少个人。
问题简单化:如果有一排人,单向传递,给出序列求解最少多少人。那么问题就是:求解一个最短序列使得给出的所有序列都是该序列的连续子序列。
再分别解决以下问题:
- 第一个人可以向左向右传递: 每个传递序列有两种情况,分别是正序与逆序。因为不清楚当前序列是向左传还是向右传。
- 传递是一个环:规定将第一个序列正向串作为首序列,累计答案的时候只需要算上重叠部分即可。
- 输入可能是绕过好几圈的情况:如果绕过好几圈,那么其他的字串因已经被包含而舍弃,dp过程不会进行,最后累计ans时ans=len[0]-重叠(s[0][0],s[0][0])即最小循环节的长度。
- 输入保证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的更多相关文章
- 【暑假】[深入动态规划]UVa 10618 Fixing the Great Wall
UVa 10618 Fixing the Great Wall 题目: http://acm.hust.edu.cn/vjudge/problem/viewProblem.action?id=361 ...
- 【暑假】[深入动态规划]UVa 10618 Tango Tango Insurrection
UVa 10618 Tango Tango Insurrection 题目: Problem A: Tango Tango Insurrection You are attempting to lea ...
- 【暑假】[深入动态规划]UVa 10618 The Bookcase
UVa 12099 The Bookcase 题目: http://acm.hust.edu.cn/vjudge/problem/viewProblem.action?id=42067 思路: ...
- 【暑假】[深入动态规划]UVa 1628 Pizza Delivery
UVa 1628 Pizza Delivery 题目: http://acm.hust.edu.cn/vjudge/problem/viewProblem.action?id=51189 思路: ...
- 【暑假】[深入动态规划]UVa 1380 A Scheduling Problem
UVa 1380 A Scheduling Problem 题目: http://acm.hust.edu.cn/vjudge/problem/viewProblem.action?id=41557 ...
- 【暑假】[深入动态规划]UVa 12170 Easy Climb
UVa 12170 Easy Climb 题目: http://acm.hust.edu.cn/vjudge/problem/viewProblem.action?id=24844 思路: 引别人一 ...
- 【暑假】[深入动态规划]UVa 1627 Team them up!
UVa 1627 Team them up! 题目: Team them up! Time Limit: 3000MS Memory Limit: Unknown 64bit IO Forma ...
- 【暑假】[深入动态规划]UVa 1412 Fund Management
UVa 1412 Fund Management 题目: UVA - 1412 Fund Management Time Limit: 3000MS Memory Limit: Unknown ...
- ACM - 动态规划 - UVA 1347 Tour
UVA 1347 Tour 题解 题目大意:有 \(n\) 个点,给出点的 \(x\).\(y\) 坐标.找出一条经过所有点一次的回路,从最左边的点出发,严格向右走,到达最右点再严格向左,回到最左点. ...
随机推荐
- python读取mnist
python读取mnist 其实就是python怎么读取binnary file mnist的结构如下,选取train-images TRAINING SET IMAGE FILE (train-im ...
- Good Bye 2015 A
Problem A:http://codeforces.com/problemset/problem/611/A A. New Year and Days 题意:某人要在2016年收集糖果,有两种不同 ...
- AndroidManifest.xml文件综合详解(转)
一,重要性AndroidManifest.xml是Android应用程序中最重要的文件之一.它是Android程序的全局配置文件,是每个 android程序中必须的文件.它位于我们开发的应用程序的根目 ...
- "Principles of Reactive Programming" 之<Actors are Distributed> (3)
Cluster 讲课的这哥们接下来讲了下Akka Cluster的使用,但是是通过把一个以前讲过的actor 系统改成使用cluster来介绍的Akka cluster. 这部分代码很多,还是直接看视 ...
- Python:使用threading模块实现多线程编程
转:http://blog.csdn.net/bravezhe/article/details/8585437 Python:使用threading模块实现多线程编程一[综述] Python这门解释性 ...
- VARCHAR2转换为CLOB碰到ORA-22858错误
近日工作中发现有一张表的字段类型建错了,本应是BLOB类型却被别人建成了VARCHAR2(200),修改时oracle却提示“ORA-22858 invalid alteration of datat ...
- C++:在程序中获取全球唯一标识号(GUID或UUID)
Windows:使用CoCreateGuid函数(GUID) #include <objbase.h> #include <stdio.h> #define GUID_LEN ...
- Head First 设计模式笔记:单例模式
单例模式 确保一个类只有一个实例,并提供一个全局访问点. 类图: Singleton static uniqueInstance //其他属性... static getInstance() //其他 ...
- EOJ-1708//POJ3334
题意: 有一个连通器,由两个漏斗组成(关于漏斗的描述见描述). 现向漏斗中注入一定量的水,问最终水的绝对位置(即y轴坐标) 思路: 总体来说分为3种情况. 1.两个漏斗可能同时装有水. 2.只可能a漏 ...
- thinkphp 定制错误页面
在前台配置文件里加上: 'TMPL_EXCEPTION_FILE' => '.Public/tpl/error.html',// 异常cuowu页面的模板文件 然后在Public下新建一个tpl ...