DNA Sequence

DNA Sequence

题意:题目给m个病毒串,问不包含病毒串的长度为n的DNA p段有几个

难度:两颗星

思路:

图片说明
这个图是例子{“ACG”,”C”},构建树后如图所示,从每个结点出发都有4条边(A,T,C,G)
•从状态0出发走一步有4种走法:
–走A到状态1(安全);
–走C到状态4(危险);
–走T到状态0(安全);
–走G到状态0(安全);
•所以当n=1时,答案就是3
•当n=2时,就是从状态0出发走2步,就形成一个长度为2的字符串,只要路径上没有经过危险结点,有几种走法,那么答案就是几种。依此类推走n步就形成长度为n的字符串。
matrix矩阵如下(矩阵i行j列的权值是结点i转移到结点j的方案数):
n=1
2 1 0 0 0
2 1 0 0 0
0 0 0 0 0
0 0 0 0 0
0 0 0 0 0
n=2
6 3 0 0 0
6 3 0 0 0
0 0 0 0 0
0 0 0 0 0
0 0 0 0 0

代码+注释:

#include<cstdio>
#include<cstring>
#include<iostream> 
#include<queue>
#define ll long long
using namespace std;
int ch[111][4],f[111],sz;
int idx[128];            //病毒只有4个字符,可以简单处理 
bool val[111];
/*题目的数据范围是10个长度10的病毒串,所以Trie树中最多101左右个节点,
那么AC自动机整个转移就可以构建一张101*101的邻接矩阵,矩阵i行j列的权值是节点i转移到节点j的方案数。
而进行k次转移,从节点i转移到节点j的方案数是这个矩阵的k次幂,这个结论离散数学的图论有(我没学,只能信了)*/
struct Trie{
    Trie(){ sz = 0; memset(ch,0,sizeof(ch)); memset(val,0,sizeof val);};
    void insert(char *s){
        int u=0;
        for(int i=0;s[i];++i){
            int c=idx[s[i]];
            if(!ch[u][c]) ch[u][c] = ++sz;
            u=ch[u][c];
        }
        val[u]=1;    //val数组标记节点是否为单词结尾,last数组和val数组是同一个数组,没必要都要,所以这里没有last数组 
    }
};
//fail指针没必要初始化,从前往后计算会覆盖上一次的结果 
int getFail(){
    queue<int>q; int u;
    f[0] = 0;
    for(int c=0;c<4;++c){
        u = ch[0][c];
        if(u)  { f[u]=0; q.push(u);}
    }
    while(!q.empty()){
        int r=q.front(); q.pop();
        for(int c=0;c<4;++c){
            u=ch[r][c];
            if(!u){
                ch[r][c]=ch[f[r]][c];    //移花接木,因为是从前往后计算的,如果当前位置没有节点,
        //而前面有这条边,则将最近的一个边的出节点复制过来,方便后续寻找fail指针的简化,只要找一次就能得到fail指针
        //因为是将前面的节点移过来,所以不必再对这个节点继续操作了 
                continue;
            }
            q.push(u);
            f[u] = ch[f[r]][c];            //可以直接这样得益于上面移花接木的操作 
            val[u]|=val[ch[f[r]][c]];    //如果该节点的fail指针是单词结尾,这个节点也不能取 
        /*只要算出后缀含有病毒的点即可,因为矩阵是通过相互到达传递的,
        其后的点只有这个危险点能到达,危险点的行和列全是0,则其后的点也不会参与运算。
        可以这样说,0行里危险点的列和其后的列从来都是0,因为不能到达。*/ 
        }
    }
}
struct mat {
    ll matrix[111][111];
    mat() {
        memset(matrix,0,sizeof(matrix));
    }
};
mat operator*(const mat &m1,const mat &m2){
    mat m;
    for(int i=0; i<=sz; ++i){
        for(int j=0; j<=sz; ++j){
            for(int k=0; k<=sz; ++k){
                m.matrix[i][j]+=m1.matrix[i][k]*m2.matrix[k][j];
                m.matrix[i][j]%=100000;
            }
        }
    }
    return m;
}
char str[11];
int m,n;
int main(){
    idx['A']=0; idx['C']=1; idx['T']=2; idx['G']=3;
    while(~scanf("%d%d",&m,&n)) {
        Trie trie;
        while(m--){
            scanf("%s",str);
            trie.insert(str);
        }
        getFail();
        mat res,a;
        for(int i=0;i<=sz;++i)    res.matrix[i][i]=1;
        for(int i=0;i<=sz;++i) {
            if(val[i])    continue;
            for(int j=0;j<4;++j) {
                if(val[ch[i][j]])    continue;
                ++a.matrix[i][ch[i][j]];
            }
        }
        for(int i=0;i<=sz;++i) {//打印矩阵
            for(int j=0;j<=sz;++j) {
                printf("%d ",a.matrix[i][j]);
            }
            puts("");
        } 
        while(n) {
            if(n&1)    res=res*a;
            a=a*a;
            n>>=1;
        }
        for(int i=0;i<=sz;++i) {//打印矩阵
            for(int j=0;j<=sz;++j) {
                printf("%d ",res.matrix[i][j]);
            }
            puts("");
        } 
        ll ans=0;
        for(int i=0;i<=sz;++i)
            ans=(ans+res.matrix[0][i])%100000;
//所有串都是从根节点开始的 
        printf("%lld\n",ans);
    }
}

无注释代码:

#include<cstdio>
#include<cstring>
#include<iostream> 
#include<queue>
#define ll long long
using namespace std;
int ch[111][4],f[111],sz;
int idx[128];
bool val[111];
struct Trie{
    Trie(){ sz = 0; memset(ch,0,sizeof(ch)); memset(val,0,sizeof val);};
    void insert(char *s){
        int u=0;
        for(int i=0;s[i];++i){
            int c=idx[s[i]];
            if(!ch[u][c]) ch[u][c] = ++sz;
            u=ch[u][c];
        }
        val[u]=1;
    }
};
int getFail(){
    queue<int>q; int u;
    f[0] = 0;
    for(int c=0;c<4;++c){
        u = ch[0][c];
        if(u)  { f[u]=0; q.push(u);}
    }
    while(!q.empty()){
        int r=q.front(); q.pop();
        for(int c=0;c<4;++c){
            u=ch[r][c];
            if(!u){
                ch[r][c]=ch[f[r]][c];
                continue;
            }
            q.push(u);
            f[u] = ch[f[r]][c];
            val[u]|=val[ch[f[r]][c]];
        }
    }
}
struct mat {
    ll matrix[111][111];
    mat() {
        memset(matrix,0,sizeof(matrix));
    }
};
mat operator*(const mat &m1,const mat &m2){
    mat m;
    for(int i=0; i<=sz; ++i){
        for(int j=0; j<=sz; ++j){
            for(int k=0; k<=sz; ++k){
                m.matrix[i][j]+=m1.matrix[i][k]*m2.matrix[k][j];
                m.matrix[i][j]%=100000;
            }
        }
    }
    return m;
}
char str[11];
int m,n;
int main(){
    idx['A']=0; idx['C']=1; idx['T']=2; idx['G']=3;
    while(~scanf("%d%d",&m,&n)) {
        Trie trie;
        while(m--){
            scanf("%s",str);
            trie.insert(str);
        }
        getFail();
        mat res,a;
        for(int i=0;i<=sz;++i)    res.matrix[i][i]=1;
        for(int i=0;i<=sz;++i) {
            if(val[i])    continue;
            for(int j=0;j<4;++j) {
                if(val[ch[i][j]]) continue;
                ++a.matrix[i][ch[i][j]];
            }
        }
        while(n) {
            if(n&1)    res=res*a;
            a=a*a;
            n>>=1;
        }
        ll ans=0;
        for(int i=0;i<=sz;++i)
            ans=(ans+res.matrix[0][i])%100000;
        printf("%lld\n",ans);
    }
}
全部评论

相关推荐

不愿透露姓名的神秘牛友
昨天 12:31
以前小时候我最痛恨出轨、偷情的人,无论男女,为什么会出轨?现在我成了自己最讨厌的人,没想到分享的东西在牛客会被这么多人看,大家的评价都很中肯,我也认同,想过一一回复,但我还是收声了,我想我应该说说这件事,这件事一直压在我心里,是个很大的心结,上面说了人为什么出轨,我大概能明白了。我们大一下半年开始恋爱,开始恋爱,我给出了我铭记3年的承诺,我对她好一辈子,我永远不会背叛,我责任心太重,我觉得跟了我,我就要照顾她一辈子,我们在一起3年我都没有碰过她,她说往东我就往东,她说什么我做什么,她要我干什么,我就干什么!在学校很美好,中途也出过一些小插曲,比如男闺蜜、男闺蜜2号等等等。但我都强迫她改掉了,我...
牛客刘北:两个缺爱的人是没有办法好好在一起的,但世界上哪有什么是非对错?你后悔你们在一起了,但是刚刚在一起的美好也是真的呀,因为其他人的出现,你开始想要了最开始的自己,你的确对不起自己,21岁的你望高物远,你完全可以不谈恋爱,去过你想要的生活,你向往自由,在一起之后,你要想的不是一个人,而是两个人,你不是变心了,就像你说的,你受够了,你不想包容了,冷静几天是你最优的选择,爱人先爱己。
社会教会你的第一课
点赞 评论 收藏
分享
深夜书店vv:腾讯是这样的,去年很多走廊都加桌子当工区
点赞 评论 收藏
分享
不愿透露姓名的神秘牛友
07-10 11:33
点赞 评论 收藏
分享
评论
2
收藏
分享

创作者周榜

更多
牛客网
牛客网在线编程
牛客网题解
牛客企业服务