zoukankan      html  css  js  c++  java
  • 【NOIP2015】子串

    题面

    有两个仅包含小写英文字母的字符串 A 和 B

    现在要从字符串 A 中取出 k 个互不重叠的非空子串,然后把这 k 个子串按照其在字符串 A 中出现的顺序依次连接起来得到一个新的字符串。请问有多少种方案可以使得这个新串与字符串 B 相等?

    注意:子串取出的位置不同也认为是不同的方案。

    分析

    定义f[i][j][p][v]表示从A串的前i位中截取p个子串,去匹配B串的前j位,v表示i这一位选还是不选
    因为同一段字符串,例如aab,可被拆成aa和b,也可被拆成a和ab,且如果都能匹配B串,则是两种方案
    所以记录i这一位是否选择的信息,可以方便A串划分子串
    转移
    a[i]==b[j]时

    f[i][j][p][0]=f[i-1][j][p][0]+f[i-1][j][p][1]
    若要在A的前i位之内匹配B的前j位,第i位又不能选,就只能在前i-1位中匹配完B的前j位

    f[i][j][p][1]=f[i-1][j-1][p][1]+f[i-1][j-1][p-1][1]+f[i-1][j-1][p-1][0]
    分为两种情况。
    大前提是,选择了第i位,则说明前i-1位只需要匹配前j位。
    1.把i这一位和第i-1位划分到一个子串内。
    所以p的数量是不变的,并且显然i-1必须被选择。-->f[i-1][j-1][p][1]
    2.把i这一位不和第i-1位划分到同一个子串内。
    则p的数量要腾出来一个,即减一。i-1这一位可选可不选。-->f[i-1][j-1][p-1][1]+f[i-1][j-1][p-1][0]

    a[i]!=b[j]时
    f[i][j][p][0]=f[i-1][j][p][0]+f[i-1][j][p][1]同理
    f[i][j][p][1]=0。因为如果不匹配还选择了这一位,只会使前面匹配的串连上这个字符后完全失效。

    空间会有200*200*1000*2=8*1e7
    所以考虑空间优化,观察到i这一位只和i-1有关系,所以将第一位滚动。

    代码

    #include<iostream>
    #include<cstdio>
    #include<cstring>
    #include<algorithm>
    using namespace std;
    #define N 1010
    #define M 210
    #define ll long long
    #define mod 1000000007
    bool v=1;
    int n,m,k;
    char a[N],b[M];
    ll f[2][M][M][2];
    int main()
    {
        scanf("%d%d%d%s%s",&n,&m,&k,a+1,b+1);
        f[0][0][0][0]=f[1][0][0][0]=1;
        for(int i=1;i<=n;i++,v^=1)
            for(int j=1;j<=m;j++)
                for(int p=1;p<=k;p++)
                {
                    f[v][j][p][0]=(f[v^1][j][p][0]+f[v^1][j][p][1])%mod;
                    if(a[i]==b[j])
                        f[v][j][p][1]=(f[v^1][j-1][p][1]+f[v^1][j-1][p-1][1]+f[v^1][j-1][p-1][0])%mod;
                    else
                        f[v][j][p][1]=0;        
                }
        printf("%lld
    ",(f[n&1][m][k][1]+f[n&1][m][k][0])%mod);
        return 0;
    }
    “Make my parents proud,and impress the girl I like.”
  • 相关阅读:
    mysql 触发器
    监听mac价格变动,降价了就通知。
    mysql 索引、约束
    mysql 三种insert插入语句
    idea 控制台中文乱码
    go 刷算法第三题——二叉树根节点到叶子节点和为指定值的路径
    mysql 排序序号sql+斐波那契统计
    go 刷算法第二题——最长回文子串
    常见排序算法-基数排序、计数排序
    常见排序算法-选择、冒泡排序
  • 原文地址:https://www.cnblogs.com/NSD-email0820/p/9734138.html
Copyright © 2011-2022 走看看