zoukankan      html  css  js  c++  java
  • [BZOJ4560][JLOI2016]字符串覆盖(贪心+DP)

    先用KMP求出所有可以放的位置,然后两个值分别处理。

    最大值:

    贪心,4!枚举放的先后位置顺序,2^3枚举相邻两个串是否有交。

    若有交,则后一个的起始位置一定是离前一个的结束位置最近的位置,无交也一样。

    最小值:

    首先去掉被其它串包含的串,因为肯定可以和其它串放同样的位置。

    将所有串从长到短排序方便DP。

    f[S][i]表示当前放的串的情况为S,串目前所覆盖到的最后一个位置为i,覆盖的最小总长度是多少,则有:

    当最后一个覆盖到i的串位置与其它串不相交时:f[S][i]=min{f[S'][k]}  这个用一个前缀和优化即可。

    当相交时:f[S][i]=min(f[S'][j]+i-j) 因为串已经从长到短排好序了所有没有问题。这个用单调队列优化即可。

    写的龟速,跑得飞快。

     1 #include<cstdio>
     2 #include<algorithm>
     3 #include<cstring>
     4 #define mem(a) memset(a,0,sizeof(a))
     5 #define rep(i,l,r) for (int i=(l); i<=(r); i++)
     6 using namespace std;
     7 
     8 const int N=10010;
     9 char s[5][N],A[N];
    10 bool b[5],fl[5][N];
    11 int T,n,len,L[5],pos[5][N],num[5],nxt[N],p[5],st[20],ed[20],f[20][N],g[20][N],q[20][N];
    12 
    13 void KMP(char S[],char T[],int pos[],int &num,bool fl[]){
    14     int n=strlen(T+1),m=strlen(S+1),j=0; nxt[1]=0;
    15     rep(i,2,m){
    16         while (j && S[j+1]!=S[i]) j=nxt[j];
    17         if (S[j+1]==S[i]) nxt[i]=++j; else nxt[i]=0; 
    18     }
    19     j=0; num=0;
    20     rep(i,1,n){
    21         while (j && S[j+1]!=T[i]) j=nxt[j];
    22         if (S[j+1]==T[i]) j++;
    23         if (j==m) pos[++num]=i-m+1,fl[i]=1,j=nxt[j];
    24     }
    25 }
    26 
    27 bool chk(char S[],char T[]){
    28     int n=strlen(T+1),m=strlen(S+1);
    29     rep(i,1,n-m+1){
    30         bool flag=0;
    31         rep(j,1,m) if (S[j]!=T[i+j-1]){ flag=1; break; }
    32         if (!flag) return 1;
    33     }
    34     return 0;
    35 }
    36 
    37 void solve_max(){
    38     rep(i,1,n) p[i]=i; int ans=0; mem(fl);
    39     do{
    40         for (int S=0; S<1<<(n-1); S++){
    41             int x=pos[p[1]][1]+L[p[1]]-1,res=L[p[1]]; ans=max(ans,res); bool flag=0;
    42             rep(i,2,n){
    43                 int d=upper_bound(pos[p[i]]+1,pos[p[i]]+num[p[i]]+1,x)-pos[p[i]];
    44                 if (S&(1<<(i-2))){
    45                      if (d<=num[p[i]] && pos[p[i]][d]>x) res+=L[p[i]],x=pos[p[i]][d]+L[p[i]]-1;
    46                     else{ flag=1; break; }
    47                 }else{
    48                     d--;
    49                     if (d && pos[p[i]][d]<=x) res+=L[p[i]]-(x-pos[p[i]][d]+1),x=pos[p[i]][d]+L[p[i]]-1;
    50                     else{ flag=1; break; }
    51                 }
    52                 if (!flag) ans=max(ans,res);
    53             }
    54         }
    55     }while (next_permutation(p+1,p+n+1));
    56     printf("%d
    ",ans);
    57 }
    58 
    59 void solve_min(){
    60     int S0=(1<<n)-1;
    61     rep(i,1,n-1) rep(j,i+1,n) if (!b[j] && chk(s[j],s[i])) b[j]=1,S0^=1<<(j-1);
    62     memset(f,0x3f,sizeof(f)); memset(g,0x3f,sizeof(g));
    63     mem(f[0]); mem(g[0]);
    64     rep(i,0,(1<<n)-1) st[i]=1,ed[i]=0;
    65     rep(i,1,len){
    66         for (int S=1; S<1<<n; S++){
    67             if (S&(S^S0)) continue; g[S][i]=g[S][i-1];
    68             rep(j,1,n) if (S&(1<<(j-1)) && fl[j][i]){
    69                 int S1=S^(1<<(j-1));
    70                 if (i-L[j]>=0) f[S][i]=min(f[S][i],g[S1][i-L[j]]+L[j]);
    71                 while (st[S1]<=ed[S1] && q[S1][st[S1]]<=i-L[j]) st[S1]++;
    72                 if (st[S1]<=ed[S1]) f[S][i]=min(f[S][i],f[S1][q[S1][st[S1]]]+i-q[S1][st[S1]]);
    73                 while (st[S]<=ed[S] && f[S][q[S][ed[S]]]-q[S][ed[S]]>=f[S][i]-i) ed[S]--;
    74                 q[S][++ed[S]]=i; g[S][i]=min(g[S][i-1],f[S][i]);
    75             }
    76         }
    77     }
    78     printf("%d ",g[S0][len]);
    79 }
    80 
    81 int main(){
    82     freopen("bzoj4560.in","r",stdin);
    83     freopen("bzoj4560.out","w",stdout);
    84     for (scanf("%d",&T); T--; ){
    85         mem(fl); mem(b); scanf("%s",A+1); len=strlen(A+1); scanf("%d",&n);
    86         rep(i,1,n) scanf("%s",s[i]+1),L[i]=strlen(s[i]+1);
    87         rep(i,1,n-1) rep(j,i+1,n) if (L[i]<L[j]) swap(s[i],s[j]),swap(L[i],L[j]);
    88         rep(i,1,n) KMP(s[i],A,pos[i],num[i],fl[i]);
    89         solve_min(); solve_max();
    90     }
    91     return 0;
    92 }
  • 相关阅读:
    开通博客园
    ios关键字
    FirstDay
    An example for pysnmp
    remove debug symbols to a seperate file
    qemu下通过gdb调试内核时 遇到 evaluation of this expression requires the program to have a function "malloc" 错误的解决办法
    关于常识与知识的思考
    基于Qemu在ubuntu上构建linux学习环境
    How to download prebuilt toolchain
    诡异的打印异常BUG
  • 原文地址:https://www.cnblogs.com/HocRiser/p/9930778.html
Copyright © 2011-2022 走看看