bzoj千题计划241:bzoj3864: Hero meet devil

http://www.lydsy.com/JudgeOnline/problem.php?id=3864

题意:

给你一个DNA序列,求有多少个长度为m的DNA序列和给定序列的LCS为0,1,2....

求LCS方式:f[i][j]=max(f[i-1][j],f[i][j-1],f[i-1][j-1]*(s[i]==t[j]))

固定了i,相邻的j的f[i][j]值最多相差1

dp[i][j] 表示长度为i的DNA序列,将“f[ |S| ][j+1]是否比f[ |S| ][j] 大1” 这个状态压缩为j的方案数

若我们知道 状态j加上一个字母k可以到状态nxt[j][k]

那么dp[i+1][nxt[j][k]]+=dp[i][j]

关键是如何求得nxt[j][k]

再一次DP

枚举所有的状态i

令f[j] 表示加上字母k之前的LCS长度,g[j]表示加上字母k之后的LCS长度

g[j]=max(g[j-1],f[j])

如果加上的字母k和原序列第j个字母匹配 g[i]=max(g[j],f[j-1]+1)

g求完后,项邻的两个g要么相等,要么相差1

再把这个状态压缩起来即可

#include<cstdio>
#include<cstring>
#include<algorithm>

using namespace std;

const int mod=1e9+7;

int m;

char ss[16];
int L,S;
int s[16];

int ch[26];

int f[16],g[16];
int nxt[1<<15][4];

int dp[2][1<<15];
int ans[16];

void pre()
{
    int len; int c[16];
    for(int i=0;i<S;++i)
    {
        memset(f,0,sizeof(f));
        for(int j=1;j<=L;++j) f[j]=f[j-1]+(i>>j-1&1);
        for(int k=0;k<4;++k)
        {
            for(int j=1;j<=L;++j)
            {
                g[j]=max(g[j-1],f[j]);
                if(s[j]==k) g[j]=max(g[j],f[j-1]+1);
            }
            nxt[i][k]=0;
            for(int j=0;j<L;++j)
                if(g[j+1]-g[j]) nxt[i][k]+=1<<j;
        }
    }
}

int count(int x)
{
    int sum=0;
    while(x)
    {
        sum+=x&1;
        x>>=1;
    }
    return sum;
}

void DP()
{
    memset(dp,0,sizeof(dp));
    int now=1,last=0;
    dp[0][0]=1;
    for(int i=1;i<=m;++i)
    {
        memset(dp[now],0,sizeof(dp[now]));
        for(int j=0;j<S;++j)
            for(int k=0;k<4;++k)
            {
                dp[now][nxt[j][k]]+=dp[last][j];
                dp[now][nxt[j][k]]-=dp[now][nxt[j][k]]>=mod ? mod : 0;
            }
        swap(now,last);
    }
    memset(ans,0,sizeof(ans));
    int t;
    for(int i=0;i<S;++i)
    {
        t=count(i);
        ans[t]+=dp[last][i];
        ans[t]-=ans[t]>=mod ? mod : 0;
    }
    for(int i=0;i<=L;++i) printf("%d\n",ans[i]);
}

int main()
{
    ch[‘A‘-‘A‘]=0;
    ch[‘C‘-‘A‘]=1;
    ch[‘G‘-‘A‘]=2;
    ch[‘T‘-‘A‘]=3;
    int T;
    scanf("%d",&T);
    while(T--)
    {
        scanf("%s",ss+1);
        scanf("%d",&m);
        L=strlen(ss+1);
        S=1<<L;
        for(int i=1;i<=L;++i) s[i]=ch[ss[i]-‘A‘];
        pre();
        DP();
    }
    return 0;
}

原文地址:https://www.cnblogs.com/TheRoadToTheGold/p/8454799.html

时间: 2024-08-02 03:37:37

bzoj千题计划241:bzoj3864: Hero meet devil的相关文章

bzoj千题计划185:bzoj1260: [CQOI2007]涂色paint

http://www.lydsy.com/JudgeOnline/problem.php?id=1260 区间DP模型 dp[l][r] 表示涂完区间[l,r]所需的最少次数 从小到大们枚举区间[l,r] 如果col[l]==col[r] dp[l][r]=min(dp[l+1][r],dp[l][r-1],dp[l+1][r-1]+1) 否则 dp[l][r]=min(dp[l][k]+dp[k+1][r]) 我还是辣鸡啊~~~~(>_<)~~~~,这种题都不能秒 #include<c

bzoj千题计划292:bzoj2244: [SDOI2011]拦截导弹

http://www.lydsy.com/JudgeOnline/problem.php?id=2244 每枚导弹成功拦截的概率 = 包含它的最长上升子序列个数/最长上升子序列总个数 pre_len [i] 表示以i结尾的最长不下降子序列的长度 pre_sum[i] 表示对应长度下的方案数 suf_len[i] 表示以i开头的最长不下降子序列长度 suf_sum[i] 表示对应长度下的方案数 若已有了这4个数组 设最长上升子序列长度=mx 那么 如果pre_len[i]+suf_len[i] -

bzoj千题计划304:bzoj3676: [Apio2014]回文串

https://www.lydsy.com/JudgeOnline/problem.php?id=3676 回文自动机模板题 4年前的APIO如今竟沦为模板,,,╮(╯▽╰)╭,唉 #include<cstdio> #include<cstring> #include<algorithm> using namespace std; #define N 300001 char ss[N]; int s[N]; int tot=1,last; int fail[N],len

bzoj千题计划106:bzoj1014 [JSOI2008]火星人prefix

http://www.lydsy.com/JudgeOnline/problem.php?id=1014 两个后缀的最长公共前缀:二分+hash 带修改带插入:splay维护 #include<cstdio> #include<cstring> #include<iostream> #define L 100001 typedef unsigned long long ULL; using namespace std; char s[L+4]; int tot,root

bzoj千题计划108:bzoj1018: [SHOI2008]堵塞的交通traffic

http://www.lydsy.com/JudgeOnline/problem.php?id=1018 关键点在于只有两行 所以一个2*m矩形连通情况只有6种 编号即对应代码中的a数组 线段树维护 用b数组表示 节点第0/1行的最右一列是否连接了右边 来 辅助 节点的合并 查询 对两个点位于矩形的位置分4种情况讨论 两点是否联通,要考虑四种情况 (以两个位置是矩形左上角和右上角为例) 1.直接联通,线段树的节点包含了这种情况,直接判断 2. 3. 4. 后三种情况需要再查询[1,l]和[r,n

bzoj千题计划109:bzoj1019: [SHOI2008]汉诺塔

http://www.lydsy.com/JudgeOnline/problem.php?id=1019 题目中问步骤数,没说最少 可以大胆猜测移动方案唯一 (真的是唯一但不会证) 设f[i][j] 表示 从i号柱子 上把j个盘子移到 g[i][j] 柱子上的步数 初始化:f[0][1]=1,g[0][1] 根据优先级决定 设三根柱子分别为0,1,2 对于每一个f[x][i], 把前i-1个移走,把第i个移走,把前i-1个移回 令y=g[x][i-1],则k=0+1+2-x-y 我们希望 把i-

bzoj千题计划111:bzoj1021: [SHOI2008]Debt 循环的债务

http://www.lydsy.com/JudgeOnline/problem.php?id=1021 如果A收到了B的1张10元,那么A绝对不会把这张10元再给C 因为这样不如B直接给C优 由此可以推出 若A欠B20元,B欠C 30元, 那么A还C20元,B还C10元最优 所以一共只有 A->BC   B->AC  C->AB AB->C  BC->A  AC->B 这6种转移情况 根据输入,我们可以知道三人最终手中有多少钱ea.eb.ec,一共有多少钱sum 设f

bzoj千题计划112:bzoj1022: [SHOI2008]小约翰的游戏John

http://www.lydsy.com/JudgeOnline/problem.php?id=1022 http://www.cnblogs.com/TheRoadToTheGold/p/6744825.html #include<cstdio> #include<iostream> using namespace std; void read(int &x) { x=0; char c=getchar(); while(!isdigit(c)) c=getchar();

bzoj千题计划113:bzoj1023: [SHOI2008]cactus仙人掌图

http://www.lydsy.com/JudgeOnline/problem.php?id=1023 dp[x] 表示以x为端点的最长链 子节点与x不在同一个环上,那就是两条最长半链长度 子节点与x在同一个环上,环形DP,单调队列优化 对于每一个环,深度最小的那个点 有可能会更新 上层节点, 所以 每一个环DP完之后,更新 dp[深度最小的点] #include<cstdio> #include<iostream> #include<algorithm> using