zoukankan      html  css  js  c++  java
  • bzoj 3059: 归途与征程

    Description

    给出一个长度为N的由小写字母’a’~’z’和’*’组成的字符串A,一个长度为M的仅由小写字母’a’~’z’组成的字符串B。一个’*’可以匹配任意多个字符(包括0个)。求在B的所有循环同构串中,有多少个能够与A匹配。
    循环同构串:就是把B的前k个字母(0<=k<M)移到结尾所得到的M个字符串。例如abc的循环同构串有abc、bca和cab。
    A与B匹配:若除了A中的’*’号可以匹配B中的任意多个字符外,其余字符一一对应,则称A与B匹配。例如a*b*c与aadbc是匹配的,其中第一个*对应ad,第二个*对应空串。

    Input

    第一行为字符串A。
    第二行为字符串B。

    Output

    输出在B的所有循环同构串中,有多少个能够与A匹配。

    将B串倍长,将A串按通配符分割,分割后预处理每段在B串的匹配位置,枚举起始位置,强制要求A串首尾与起始、终止位置匹配,中间的段贪心取最左的,时间复杂度$O(nm)$
    #include<cstdio>
    #include<cstring>
    char s1[107],s2[210007],*ss[107];
    int l1,l2,sp=0,ls[107];
    int nx[107][200007],ans=0;
    int main(){
        scanf("%s%s",s1+1,s2+1);
        l1=strlen(s1+1);
        l2=strlen(s2+1);
        memcpy(s2+l2+1,s2+1,l2);
        s1[0]=s1[l1+1]='*';
        for(int i=1;i<=l1+1;++i)if(s1[i-1]=='*'){
            ss[sp]=s1+i;
            while(ss[sp][ls[sp]]!='*')++ls[sp];
            ++sp;
        }
        --sp;
        for(int i=0;i<=sp;++i){
            nx[i][l2*2]=l2*2;
            for(int j=l2*2-1;j;--j){
                nx[i][j]=memcmp(ss[i],s2+j,ls[i])?nx[i][j+1]:j;
            }
        }
        for(int i=1;i<=l2;++i){
            int mx=i+l2-ls[sp];
            if(mx>0&&nx[0][i]==i&&nx[sp][mx]==mx){
                int w=i;
                for(int i=0;i<sp&&w<=mx;++i)w=nx[i][w]+ls[i];
                ans+=(w<=mx);
            }
        }
        return printf("%d
    ",ans),0;
    }
  • 相关阅读:
    字符,字节和编码
    Linux网络参数和ifconfig
    默认网关 网关 子网掩码 广播地址
    S.M.A.R.T.记录几块ssd硬盘
    linux 别名
    echo 输出颜色
    Linux:echo命令详解
    centos下安装mongodb 通过shell脚本
    linux查看登录用户
    linux wget指定下载目录和重命名
  • 原文地址:https://www.cnblogs.com/ccz181078/p/6755173.html
Copyright © 2011-2022 走看看