【暑假】[深入动态规划]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\) 坐标.找出一条经过所有点一次的回路,从最左边的点出发,严格向右走,到达最右点再严格向左,回到最左点. ...
随机推荐
- linux压缩文件(夹) zip uzip命令的用法
压缩文件(夹) # 压缩列举的文件,格式如下: zip 压缩包名称 文件1 文件2 文件3 ... # 压缩test.txt, a.out文件,并取名为abc.zip $ zip abc.zip te ...
- iphone6S“玫瑰金”的秘密——阳极氧化
阳极氧化对多数人来说是一个熟悉又陌生的名词,大多数可能知道它的作用之一就是是能使金属呈现各种各样色彩.最为人熟知的运用阳极氧化技术的产品就是iphone系列产品了,已经推出了金色,玫瑰金色,深空灰色, ...
- 113. Path Sum II
题目: Given a binary tree and a sum, find all root-to-leaf paths where each path's sum equals the give ...
- POJ1850——Code(组合数学)
Code DescriptionTransmitting and memorizing information is a task that requires different coding sys ...
- Oracle10g 回收站及彻底删除table : drop table xx purge
drop后的表被放在回收站(user_recyclebin)里,而不是直接删除掉.这样,回收站里的表信息就可以被恢复,或彻底清除. 1.通过查询回收站user_recyclebin获取被删除的表信息, ...
- margin,border,padding简介
站在图中心 Content 的角度理解: margin为外边框,border为边框,padding为内边框. 在xml中设置: 如果上下左右的距离都是相同可以通过 android:layout_mar ...
- NFC(5)编写NFC程序的基本步骤
1,设置权限 <uses-permission android:name="android.permission.NFC" /> 2,限制Android版本 <u ...
- poj 1129 Channel Allocation ( dfs )
题目:http://poj.org/problem?id=1129 题意:求最小m,使平面图能染成m色,相邻两块不同色由四色定理可知顶点最多需要4种颜色即可.我们于是从1开始试到3即可. #inclu ...
- (转载)Let's Play Games!
第1题 Alice和她的同学Bob通过网上聊天商量明天早晨谁去教室打扫卫生的事,Bob说:“我在桌上放了一枚硬币,你猜一下,是正面朝上还是反面朝上?如果猜对了,我去扫地.如果猜错了,嘿嘿….” Al ...
- 三个流行MySQL分支的对比
MySQL是历史上最受欢迎的免费开源程序之一.它是成千上万个网站的数据库骨干,并且可以将它(和Linux)作为过去10年里Internet呈指数级增长的一个有力证明. 那么,如果MySQL真的这么重要 ...