【bzoj3864】Hero meet devil 题解

题目大意

  给你一个只由 AGCT 组成的字符串 S S S,对于每个 0 ≤ i ≤ ∣ S ∣ 0 \leq i \leq |S| 0iS,问有多少个只由 AGCT 组成的长度为 m m m 的字符串 T T T,使得 L C S ( S , T ) = i LCS(S,T)=i LCS(S,T)=i

   ∣ S ∣ ≤ 15 ,   m ≤ 1000 |S| \leq 15,~m \leq 1000 S15, m1000

\\
\\
\\

题解

  直到大学了才初见 dp 套 dp 呢。。这该是有多菜呢?

  假设已经拥有了一个 T T T,则可以 O ( n m ) O(nm) O(nm) 求出 L C S LCS LCS

L C S i , j = max ⁡ { L C S i − 1 , j ,   L C S i , j − 1 ,   L C S i − 1 , j − 1 + 1 ( S i = = T j ) } LCS_{i,j}=\max\{LCS_{i-1,j},~LCS_{i,j-1},~LCS_{i-1,j-1}+1(S_i==T_j)\} LCSi,j=max{LCSi1,j, LCSi,j1, LCSi1,j1+1(Si==Tj)}

  并且,对于 T T T 的某一位(假设第 j j j 位)的 dp,相邻的两个 i i i 所对应的 L C S i , j LCS_{i,j} LCSi,j 最多相差 1 1 1。也就是说,对于某一位 j j j,由 L C S 1 , j ,   L C S 2 , j ,   . . .   ,   L C S n , j LCS_{1,j},~LCS_{2,j},~...~,~LCS_{n,j} LCS1,j, LCS2,j, ... , LCSn,j 构成的 dp 状态最多 2 15 2^{15} 215 种。

  现在没有 T T T,则 dp 这个 T。设 f j , s f_{j,s} fj,s 表示到了 T T T 的第 j j j 位,dp 状态为 s s s,的方案数。预处理各方案的转移 T s , k T_{s,k} Ts,k k ∈ { A , T , C , G } k\in\{A,T,C,G\} k{A,T,C,G}),就可以 O ( m ⋅ 2 ∣ S ∣ ) O(m\cdot2^{|S|}) O(m2S) dp 了。

代码

#include<bits/stdc++.h>
#define fo(i,a,b) for(int i=a;i<=b;i++)
#define fd(i,a,b) for(int i=a;i>=b;i--)
using namespace std;

typedef long long LL;

const int maxm=1005, maxN=32800;
const LL mo=1e9+7;

int n,m,N,er[20];
char S[20];

LL f[2][maxN];
int Q,fn[maxm][2],T[maxN][4],cnt[maxN];
char tb[4]={'A','G','C','T'};
int main()
{
	fo(i,0,14) er[i]=1<<i;
	
	scanf("%d",&Q);
	while (Q--)
	{
		scanf("%s",S+1);
		n=strlen(S+1);
		scanf("%d",&m);
		
		N=(1<<n)-1;
		fo(s,0,N)
		{
			fo(k,0,3)
			{
				fo(i,1,n) fn[i][0]=fn[i-1][0]+((s&er[n-i])>0);
				fo(i,1,n) fn[i][1]=max(max(fn[i-1][1],fn[i][0]),(S[i]==tb[k])?fn[i-1][0]+1:0);
				int news=0;
				fd(i,n,1) fn[i][1]-=fn[i-1][1], news|=(fn[i][1]<<(n-i));
				T[s][k]=news;
				cnt[s]=fn[n][0];
			}
		}
		
		memset(f,0,sizeof(f));
		f[0][0]=1;
		int now=0;
		fo(i,0,m-1)
		{
			memset(f[now^1],0,sizeof(f[now^1]));
			fo(s,0,N)
				fo(k,0,3)
				{
					LL *p=&f[now^1][T[s][k]];
					*p+=f[now][s];
					*p=(*p>=mo) ?*p-mo :*p ;
				}
			now^=1;
		}
		
		fo(i,0,n)
		{
			LL ans=0;
			fo(s,0,N) if (cnt[s]==i) (ans+=f[now][s])%=mo;
			printf("%lld\n",ans);
		}
	}
}
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值