BZOJ4560 [JLoi2016]字符串覆盖
题意
字符串A有N个子串B1,B2,…,Bn。如果将这n个子串分别放在恰好一个它在A中出现的位置上(子串之间可以重叠)
这样A中的若干字符就被这N个子串覆盖了。问A中能被覆盖字符个数的最小值和最大值。
字符串长度\(A<=10000,N<=4\),子串长充\(<=1000\)
分析
参照_Gloid的题解。
首先kmp求出每个子串能放在哪些位置。接下来的两部分贪心和状压都可以,各取比较方便的。
最大值
考虑贪心。考虑枚举子串的左端点出现顺序,在此基础上每个子串的位置肯定都应该尽量靠前,有是否与上个子串有交两种选择,如果有交一定会使交集最小,于是枚举第一个子串出现位置并暴力枚举\(4!*2^3\)种情况。时间复杂度\(O(N! 2^{N-1})\),随便跑。
最小值
考虑状压。首先把被包含的子串去掉方便处理。将线段排序,设\(f[i][S]\)考虑前\(i\)条线段已出现的子串集合为\(S\)时的最小覆盖长度,转移时考虑上条线段是否与其有交,单调队列优化转移(因为懒写了线段树)。虽然非常麻烦但可能还是比贪心好点的。而最大值由于不能删掉被包含子串状压简直没法做。
关于线段树存的内容的含义,线段树处理线段相交的情况,\(dp[i]=dp[j]+i-j\)这种形式的DP,可以先把\(dp[j]-j\)存下来,然后读取的时候直接读该位置的最大/最小值\(+i\)即可。这里存的就是\(f[i][S]-r[i]\)。
时间复杂度\(O(A 2^N \log_2 A)\),是2126033.9807279119026370044348732。
代码
利用函数入栈顺序简化了代码。话说我之前都是被这个入栈顺序坑了……
#include<bits/stdc++.h>
#define rg register
#define il inline
#define co const
template<class T>il T read()
{
rg T data=0;
rg int w=1;
rg char ch=getchar();
while(!isdigit(ch))
{
if(ch=='-')
w=-1;
ch=getchar();
}
while(isdigit(ch))
{
data=data*10+ch-'0';
ch=getchar();
}
return data*w;
}
template<class T>il T read(rg T&x)
{
return x=read<T>();
}
typedef long long ll;
using namespace std;
co int N=1e4+1;
char s[N],t[5][N];
int n,m,len[5],cnt[5];
int fail[N];
struct data
{
int l,r,id;
bool operator<(co data&x)co
{
return l<x.l;
}
}b[5][N],c[4*N];
namespace MAX
{
bool flag[5];
int ans;
void dfs(int num,int pl,int pr,int sum)
{
if(num==m)
{
ans=std::max(ans,sum);
return;
}
for(int i=1;i<=m;++i)if(!flag[i])
{
int u=0,v=0;
for(int j=1;j<=cnt[i];++j)if(b[i][j].l>=pl)
{
if(b[i][j].l<=pr) u=j;
else {v=j;break;}
}
flag[i]=1;
if(u)
dfs(num+1,b[i][u].l,std::max(pr,b[i][u].r),sum+std::max(b[i][u].r-pr,0));
if(v)
dfs(num+1,b[i][v].l,std::max(pr,b[i][v].r),sum+b[i][v].r-b[i][v].l+1);
flag[i]=0;
}
}
int solve()
{
// cerr<<"solve max"<<endl;
ans=0;
dfs(0,0,0,0);
return ans;
}
}
namespace MIN
{
int min[N*4][1<<4],f[N*4][1<<4],tree[N<<4][1<<4];
void rebuild()
{
bool flag[5]={0};
for(int i=1;i<=m;++i)
for(int j=1;j<=m;++j)
if(i!=j&&len[i]<len[j]||len[i]==len[j]&&i>j)
for(int k=1;k<=cnt[i];++k)
if(b[i][k].l>=b[j][1].l&&b[i][k].r<=b[j][1].r) {flag[i]=1;break;}
n=0;int m2=0;
for(int i=1;i<=m;++i)if(!flag[i])
{
for(int j=1;j<=cnt[i];++j)
c[++n]=b[i][j],c[n].id=m2;
++m2;
}
m=m2;
sort(c+1,c+n+1);
}
void insert(int x,int s,int l,int r,int p,int v)
{
tree[x][s]=std::min(tree[x][s],v);
if(l==r)
return;
int m=(l+r)/2;
if(p<=m)
insert(x<<1,s,l,m,p,v);
else
insert(x<<1|1,s,m+1,r,p,v);
}
int query(int x,int s,int l,int r,int ql,int qr)
{
if(ql>qr)
return N;
if(ql<=l&&r<=qr)
return tree[x][s];
int m=(l+r)/2;
if(qr<=m)
return query(x<<1,s,l,m,ql,qr);
if(ql>=m+1)
return query(x<<1|1,s,m+1,r,ql,qr);
return std::min(query(x<<1,s,l,m,ql,qr),query(x<<1|1,s,m+1,r,ql,qr));
}
int solve()
{
// cerr<<"solve min"<<endl;
rebuild();
memset(f,0x3f,sizeof f);f[0][0]=0;
memset(min,0x3f,sizeof min);min[0][0]=0;
memset(tree,0x3f,sizeof tree);
int p=0;
for(int i=1;i<=n;++i)
{
while(c[p+1].r<c[i].l) ++p;
for(int j=0;j<(1<<m);++j)if(j&(1<<c[i].id))
f[i][j]=std::min(min[p][j^(1<<c[i].id)]+c[i].r-c[i].l+1,query(1,j^(1<<c[i].id),1,n,p+1,i-1)+c[i].r);
for(int j=0;j<(1<<m);++j)
{
min[i][j]=std::min(min[i-1][j],f[i][j]);
insert(1,j,1,n,i,f[i][j]-c[i].r);
}
}
return min[n][(1<<m)-1];
}
}
int main()
{
// freopen(".in","r",stdin);
// freopen(".out","w",stdout);
int kase;
read(kase);
while(kase--)
{
scanf("%s",s+1);
n=strlen(s+1);
read(m);
for(int i=1;i<=m;++i)
{
scanf("%s",t[i]+1);
len[i]=strlen(t[i]+1),cnt[i]=0;
for(int j=1;j<len[i];++j)
{
int k=fail[j];
while(k&&t[i][k+1]!=t[i][j+1])
k=fail[k];
fail[j+1]=t[i][k+1]==t[i][j+1]?k+1:0;
}
int j=0;
for(int k=1;k<=n;++k)
{
while(j&&t[i][j+1]!=s[k])
j=fail[j];
if(t[i][j+1]==s[k])
++j;
if(j==len[i])
++cnt[i],b[i][cnt[i]].l=k-len[i]+1,b[i][cnt[i]].r=k;
}
sort(b[i]+1,b[i]+cnt[i]+1);
// for(int j=1;j<=cnt[i];++j)
// cerr<<j<<" l="<<b[i][j].l<<" r="<<b[i][j].r<<endl;
}
printf("%d %d\n",MIN::solve(),MAX::solve());
}
return 0;
}
BZOJ4560 [JLoi2016]字符串覆盖的更多相关文章
- [BZOJ4560][JLOI2016]字符串覆盖(贪心+DP)
先用KMP求出所有可以放的位置,然后两个值分别处理. 最大值: 贪心,4!枚举放的先后位置顺序,2^3枚举相邻两个串是否有交. 若有交,则后一个的起始位置一定是离前一个的结束位置最近的位置,无交也一样 ...
- BZOJ4560 JLOI2016字符串覆盖(kmp+贪心+状压dp+单调队列)
首先kmp求出每个子串能放在哪些位置.接下来的两部分贪心和状压都可以,各取比较方便的. 最大值考虑贪心.考虑枚举子串的左端点出现顺序,在此基础上每个子串的位置肯定都应该尽量靠前,有是否与上个子串有交两 ...
- 【BZOJ4560】[JLoi2016]字符串覆盖 KMP+状压DP
[BZOJ4560][JLoi2016]字符串覆盖 Description 字符串A有N个子串B1,B2,…,Bn.如果将这n个子串分别放在恰好一个它在A中出现的位置上(子串之间可以重叠)这样A中的若 ...
- 并不对劲的bzoj4560:p3269:[JLOI2016]字符串覆盖
题目大意 \(T\)(\(T\leq10\))组询问 每组询问给出一个字符串\(A\)(\(|A|\leq10^4\)),\(n\)(\(n\leq4\))个\(A\)的子串\(B_1,B_2,B_3 ...
- 【BZOJ】4560: [JLoi2016]字符串覆盖
题解 先用kmp求出来一个ed[i][j]表示在母串的第i位是第j个子串的结尾 考虑状压一个二进制位表示这个子串覆盖过没有 对于最大值,记一个dp[S][i]表示子串的使用状况为S,当前为母串的第i位 ...
- 笔记:iOS字符串的各种用法(字符串插入、字符串覆盖、字符串截取、分割字符串)(别人的代码直接复制过来的,我脸皮有点厚)
NSString* str=@"hello";//存在代码区,不可变 NSLog(@"%@",str); //1.[字符串插入] NSMutableString ...
- iOS字符串的各种用法(字符串插入、字符串覆盖、字符串截取、分割字符串)
NSString* str=@"hello";//存在代码区,不可变 NSLog(@"%@",str); //1.[字符串插入] NSMutableString ...
- bzoj AC倒序
Search GO 说明:输入题号直接进入相应题目,如需搜索含数字的题目,请在关键词前加单引号 Problem ID Title Source AC Submit Y 1000 A+B Problem ...
- 贪心/构造/DP 杂题选做Ⅱ
由于换了台电脑,而我的贪心 & 构造能力依然很拉跨,所以决定再开一个坑( 前传: 贪心/构造/DP 杂题选做 u1s1 我预感还有Ⅲ(欸,这不是我在多项式Ⅱ中说过的原话吗) 24. P5912 ...
随机推荐
- Java:判断字符串中包含某字符的个数
Java:判断字符串中包含某字符的个数 JAVA中查询一个词在内容中出现的次数: public int getCount(String str,String key){ if(str == null ...
- Python中用format函数格式化字符串的用法(2.7版本讲解哦!)
语法 它通过{}和:来代替%.“映射”示例 通过位置 In [1]: '{0},{1}'.format('kzc',18) Out[1]: 'kzc,18' In [2]: '{},{}'.forma ...
- [C++] 2017联发科技杯编程挑战赛 复赛题 “杰克船长的烦恼”
题目如下. 规则 杰克船长这次运气不错,抢到了一大堆金币.但他马上又开始发愁了, 因为如何给大家分金币,一直都是件不容易的事,每次杰克船长都要头疼好几天. 关于分金币,海盗的行规是这样的: 每次行动, ...
- react-native navigation的学习与使用
在很久之前,RN中文网说推荐用react-navigation替代navigator作为新的导航库,从RN 0.43版本开始,官方就已经停止维护Navigator了,所以强烈建议大家迁移到新的reac ...
- MySQL二进制日志文件过期天数设置说明
今天在处理业务库中二进制文件的时候,想更改二进制文件的过期天数,发现日期如果设置成2位以上的整数.都会出现如下的警告.不能成功的设置过期日期天数.MySQL版本从5.1到5.5都是一样的. mysql ...
- centos 6的LAMP一键安装包(可选择/升级版本)
安装步骤 事前准备(安装 wget.screen.unzip,创建 screen 会话) yum -y install wget screen git git clone 并赋予脚本执行权限 git ...
- Asp.net Core, 基于 claims 实现权限验证 - 引导篇
什么是Claims? 这个直接阅读其他大神些的文章吧,解释得更好. 相关文章阅读: http://www.cnblogs.com/JustRun1983/p/4708176.html http://w ...
- BZOJ-5424: 烧桥计划(单调队列)
BZOJ-5424: 烧桥计划(单调队列) 题目链接 题解: 先考虑最暴力的\(dp\):设\(f[k][i]\)表示搞掉第\(1\sim i\)段,烧了\(k\)段的最小花费,设\(calc(x,y ...
- Logstash过滤器修改数据
数据修改(Mutate) filters/mutate 插件是 Logstash 另一个重要插件.它提供了丰富的基础类型数据处理能力.包括类型转换,字符串处理和字段处理等. 类型转换 类型转换是 fi ...
- 一个较好的style与ControlTemplate结合的示例(以Button为例)
<!--按钮背景画刷--> <LinearGradientBrush x:Key="buttonBackgroundBrush"> &l ...