首先kmp求出每个子串能放在哪些位置。接下来的两部分贪心和状压都可以,各取比较方便的。

  最大值考虑贪心。考虑枚举子串的左端点出现顺序,在此基础上每个子串的位置肯定都应该尽量靠前,有是否与上个子串有交两种选择,如果有交一定会使交集最小,于是枚举第一个子串出现位置并暴力枚举4!*23种情况。

  最小值考虑状压。首先把被包含的子串去掉方便处理。将线段排序,设f[i][S]为当前覆盖到的最右位置为i已出现的子串集合为S时的最小覆盖长度,转移时考虑上条线段是否与其有交,单调队列优化转移(因为懒写了线段树)。虽然非常麻烦但可能还是比贪心好点的。而最大值由于不能删掉被包含子串状压简直没法做。

#include<iostream>
#include<cstdio>
#include<cmath>
#include<cstdlib>
#include<cstring>
#include<algorithm>
using namespace std;
#define ll long long
#define N 10010
int read()
{
int x=,f=;char c=getchar();
while (c<''||c>'') {if (c=='-') f=-;c=getchar();}
while (c>=''&&c<='') x=(x<<)+(x<<)+(c^),c=getchar();
return x*f;
}
char getc(){char c=getchar();while ((c<'A'||c>'Z')&&(c<'a'||c>'z')) c=getchar();return c;}
int gcd(int n,int m){return m==?n:gcd(m,n%m);}
int T,n,m,nxt[N],cnt[],mn[N*][<<],f[N*][<<],tree[<<][N<<],ans;
bool flag[];
char s[N],a[][N];
struct data
{
int x,y,op;
bool operator <(const data&a) const
{
return x<a.x;
}
}b[][N],c[*N];
void dfs(int k,int last,int r,int s)
{
if (k==m) {ans=max(ans,s);return;}
for (int i=;i<=m;i++)
if (!flag[i])
{
int u=,v=;
for (int j=;j<=cnt[i];j++)
if (b[i][j].x>=last)
if (b[i][j].x<=r) u=j;
else {v=j;break;}
flag[i]=;
if (u) dfs(k+,b[i][u].x,max(r,b[i][u].y),s+max(b[i][u].y-r,));
if (v) dfs(k+,b[i][v].x,max(r,b[i][v].y),s+b[i][v].y-b[i][v].x+);
flag[i]=;
}
}
void rebuild()
{
bool flag[]={};
for (int i=;i<=m;i++)
for (int j=;j<=m;j++)
if (i!=j&&(strlen(a[i]+)<strlen(a[j]+)||(strlen(a[i]+)==strlen(a[j]+)&&i>j)))
{
for (int k=;k<=cnt[i];k++)
if (b[i][k].x>=b[j][].x&&b[i][k].y<=b[j][].y) {flag[i]=;break;}
}
n=;int m2=;
for (int i=;i<=m;i++)
if (!flag[i])
{
for (int j=;j<=cnt[i];j++)
c[++n]=b[i][j],c[n].op=m2;
m2++;
}
m=m2;
sort(c+,c+n+);
}
void ins(int op,int k,int l,int r,int p,int x)
{
tree[op][k]=min(tree[op][k],x);
if (l==r) return;
int mid=l+r>>;
if (p<=mid) ins(op,k<<,l,mid,p,x);
else ins(op,k<<|,mid+,r,p,x);
}
int query(int op,int k,int l,int r,int x,int y)
{
if (x>y) return N;
if (l==x&&r==y) return tree[op][k];
int mid=l+r>>;
if (y<=mid) return query(op,k<<,l,mid,x,y);
else if (x>mid) return query(op,k<<|,mid+,r,x,y);
else return min(query(op,k<<,l,mid,x,mid),query(op,k<<|,mid+,r,mid+,y));
}
void work()
{
memset(f,,sizeof(f));f[][]=;
memset(mn,,sizeof(mn));mn[][]=;
memset(tree,,sizeof(tree));
int t=;
for (int i=;i<=n;i++)
{
while (c[t+].y<c[i].x) t++;
for (int j=;j<(<<m);j++)
if (j&(<<c[i].op)) f[i][j]=min(mn[t][j^(<<c[i].op)]+c[i].y-c[i].x+,query(j^(<<c[i].op),,,n,t+,i-)+c[i].y);
for (int j=;j<(<<m);j++)
mn[i][j]=min(mn[i-][j],f[i][j]),ins(j,,,n,i,f[i][j]-c[i].y);
}
ans=mn[n][(<<m)-];
}
int main()
{
#ifndef ONLINE_JUDGE
freopen("bzoj4560.in","r",stdin);
freopen("bzoj4560.out","w",stdout);
const char LL[]="%I64d\n";
#else
const char LL[]="%lld\n";
#endif
T=read();
while (T--)
{
scanf("%s",s+);n=strlen(s+);
m=read();
for (int i=;i<=m;i++) scanf("%s",a[i]+);
memset(cnt,,sizeof(cnt));
for (int i=;i<=m;i++)
{
nxt[]=-;int t=strlen(a[i]+);
for (int k=;k<=t;k++)
{
int j=nxt[k-];
while (~j&&a[i][j+]!=a[i][k]) j=nxt[j];
nxt[k]=j+;
}
int x=;
for (int k=;k<=n;k++)
{
while (~x&&a[i][x+]!=s[k]) x=nxt[x];
x++;if (x==t) cnt[i]++,b[i][cnt[i]].x=k-t+,b[i][cnt[i]].y=k,x=nxt[x];
}
}
for (int i=;i<=m;i++) sort(b[i]+,b[i]+cnt[i]+);
ans=;dfs(,,,);int tmp=ans;
rebuild();work();
cout<<ans<<' '<<max(ans,tmp)<<endl;
}
return ;
}

BZOJ4560 JLOI2016字符串覆盖(kmp+贪心+状压dp+单调队列)的更多相关文章

  1. 【BZOJ4560】[JLoi2016]字符串覆盖 KMP+状压DP

    [BZOJ4560][JLoi2016]字符串覆盖 Description 字符串A有N个子串B1,B2,…,Bn.如果将这n个子串分别放在恰好一个它在A中出现的位置上(子串之间可以重叠)这样A中的若 ...

  2. POJ 1795 DNA Laboratory (贪心+状压DP)

    题意:给定 n 个 字符串,让你构造出一个最短,字典序最小的字符串,包括这 n 个字符串. 析:首先使用状压DP,是很容易看出来的,dp[s][i] 表示已经满足 s 集合的字符串以 第 i 个字符串 ...

  3. bzoj3717 [PA2014]Pakowanie 贪心+状压DP

    题目传送门 https://lydsy.com/JudgeOnline/problem.php?id=3717 题解 这道题大概也就只能算常规的状压 DP 吧,但是这个状态和转移的设计还是不是很好想. ...

  4. [BZOJ4560][JLOI2016]字符串覆盖(贪心+DP)

    先用KMP求出所有可以放的位置,然后两个值分别处理. 最大值: 贪心,4!枚举放的先后位置顺序,2^3枚举相邻两个串是否有交. 若有交,则后一个的起始位置一定是离前一个的结束位置最近的位置,无交也一样 ...

  5. BZOJ4560 [JLoi2016]字符串覆盖

    题意 字符串A有N个子串B1,B2,-,Bn.如果将这n个子串分别放在恰好一个它在A中出现的位置上(子串之间可以重叠) 这样A中的若干字符就被这N个子串覆盖了.问A中能被覆盖字符个数的最小值和最大值. ...

  6. CodeForces165E 位运算 贪心 + 状压dp

    http://codeforces.com/problemset/problem/165/E 题意 两个整数 x 和 y 是 兼容的,如果它们的位运算 "AND" 结果等于 0,亦 ...

  7. BZOJ4416 [Shoi2013]阶乘字符串 【序列自动机 + 状压dp】

    题目链接 BZOJ4416 题解 建立序列自动机,即预处理数组\(nxt[i][j]\)表示\(i\)位置之后下一个\(j\)出现的位置 设\(f[i]\)表示合法字符集合为\(i\)的最短前缀,枚举 ...

  8. 2018.08.29 NOIP模拟 movie(状压dp/随机化贪心)

    [描述] 小石头喜欢看电影,选择有 N 部电影可供选择,每一部电影会在一天的不同时段播 放.他希望连续看 L 分钟的电影.因为电影院是他家开的,所以他可以在一部电影播放过程中任何时间进入或退出,当然他 ...

  9. Codeforces 429C Guess the Tree(状压DP+贪心)

    吐槽:这道题真心坑...做了一整天,我太蒻了... 题意 构造一棵 $ n $ 个节点的树,要求满足以下条件: 每个非叶子节点至少包含2个儿子: 以节点 $ i $ 为根的子树中必须包含 $ c_i ...

随机推荐

  1. Asp.Net Core使用Nginx实现反向代理

    ---恢复内容开始--- 前两篇文章介绍了使用Docker作为运行环境,用Nginx做反向代理的部署方法,这篇介绍一下使用Nginx配合.Net Core运行时直接在Liunx上裸奔的方法. 一.安装 ...

  2. sqlite3 简单实用方法

    打开数据库:sqlite3.exe test.db 显示所有表: .tables 退出 sqlite3:.quit 还有个问题,已经打开一个数据库文件了. 不知道如何在不退出命令行的情况下,更换另一个 ...

  3. Hadoop==zookeeper

    Zookeeper 每一个专业的技术总可以在生活中找到相应的实例,就比如说zookeeper,攘其外必先安其内就很好的解释了zookeeper,Hadoop集群的组件中的很多在学习的时候都会觉得每一个 ...

  4. pycharm中每次创建py文件时就自动生成代码头,以及出现SyntaxError:Non-ASCII 。。。问题

    我们在pycharm中执行py文件的时候,可能会出现以下错误 这是因为你没有制定编码格式,这时候你需要在文件最开始制定编码格式,代码如下 #!/user/bin/env python #-*- cod ...

  5. java基础day05---界面

    java基础day05---界面 1.GUI:图形用户界面(Graphics User Interface) 开发工具包AWT抽象窗口把工具箱===>swing 解决了awt存在的lcd问题== ...

  6. shell -- if参数用法

    一.概要1.if与[之间要有空格2.[]与判断条件之间也必须有空格3.]与:之间不能有空格 二.对字符串的判断1.if [ str1=str2 ];then fi  ----当两个字符串相同时返回真2 ...

  7. 人工智能,图片识别,与GUI编程

    GUI编程: https://sourceforge.net/projects/pyqt/ 百度aip图片识别 https://pypi.python.org/pypi/baidu-aip

  8. FJWC 2019 游记

    FJWC 2019 游记 Day 0 春节旅游, 刚从杭州绍兴一带赶回来, 然而并没有直接飞去福州, 去了厦门再去福州, 浪费了好多时间. Day 1 酒店到学校有 \(20\) 分钟的步行路程, 感 ...

  9. 【实用】如何将sublime text 3 打造成实用的python IDE 环境

    前段时间写脚本,一直使用的是pycharm ,无奈机器不配置实在不怎么样,我记得之前用过subline text,这是我用过的最酷炫的文本编辑器,参考了一下网上的文章,自己走了一些弯路,将心得写在这里 ...

  10. OpenCV入门:(五:更改图片对比度和亮度)

    1. 理论 图片的转换就是将图片中的每个像素点经过一定的变换,得到新像素点,新像素点组合成一张新的图片. 改变图片对比度和亮度的变换如下: 其中α和β被称作增益参数(gain parameter)和偏 ...