zoukankan      html  css  js  c++  java
  • BZOJ 3864 Hero Meets Devil

    题目大意

    给定一个由AGCT组成的串(t), 求对于所有的(L in [1, |t|]), 有多少个由AGCT组成的串(s)满足(LCS(s, t) = L).

    Solution

    传说中的DP套DP.

    我们用(f_{i, j})表示(s)的前(i)位与(t)的前(j)位的最长公共子序列, 则我们有

    [f_{i, j} = max egin{cases} f_{i - 1, j} \ f_{i, j - 1} \ f_{i - 1, j - 1} + [s_i = t_j] end{cases} ]

    (LCS(s, t) = f[|S|][|T|]).

    逐位考虑(s), 假设当前到(s)的第(i)位, 我们用(F)表示状态: (F = { f[i][1], f[i][2], cdots, f[i][|t|] }). 考虑当(i)变成(i + 1)时, (F)会怎么变化: 对于确定的(t), (F)的变化只与(s[i + 1])有关, 因此我们令(c =s[i + 1]), 用(T(F, c))表示当(s[i + 1] = c)(F)会变成怎么样.

    我们用(g[i][F])表示(s)的前(i)位与(t)的最长公共子序列状态为(F)的串(s)的数量, 则对于每一个(c), 有(g[i + 1][T(F, c)] += g[i][F]).

    我们发现(F)不容易被记录, 又因为注意到(0 le F[i] - F[i - 1] le 1)(F[0] = 0), 因此我们用(S)来表示({ F[i] - F[i - 1] })的状态集合. 这样我们就可以轻易地进行状态压缩了.

    考虑怎么统计答案:

    [ans[k] = sum_{F} g[|T|][F] imes [F[|T|] = k] ]

    #include <cstdio>
    #include <cstring>
    #include <algorithm>
    
    using namespace std;
    const int N = 15, MOD = (int)1e9 + 7;
    int t[N + 7];
    int n, m;
    int main()
    {
    
    #ifndef ONLINE_JUDGE
    
    	freopen("gene.in", "r", stdin);
    	freopen("gene.out", "w", stdout);
    	
    #endif
    
    	int T; scanf("%d
    ", &T);
    	for (int cs = 0; cs < T; ++ cs)
    	{
    		static char str[N + 7]; scanf("%s", str + 1);
    		n = strlen(str + 1); scanf("%d
    ", &m);
    		for (int i = 1; i <= n; ++ i)
    			if (str[i] == 'A') t[i] = 1;
    			else if (str[i] == 'T') t[i] = 2;
    			else if (str[i] == 'G') t[i] = 3;
    			else if (str[i] == 'C') t[i] = 4;
    		static int trans[(1 << N) + 7][7];
    		for (int j = 0; j < 1 << n; ++ j) for (int c = 1; c <= 4; ++ c)
    		{
    			static int a[N + 7], b[N + 7];
    			memset(a, 0, sizeof a);
    			for (int k = 0; k < n; ++ k) a[k + 1] = a[k] + (j >> k & 1);
    			b[0] = 0;
    			for (int k = 1; k <= n; ++ k) b[k] = max(max(a[k], b[k - 1]), a[k - 1] + (c == t[k]));
    			int stt = 0;
    			for (int k = 0; k < n; ++ k) if (b[k + 1] - b[k]) stt |= 1 << k;
    			trans[j][c] = stt;
    		}
    		static int f[(1 << N) + 7], g[(1 << N) + 7];
    		memset(f, 0, sizeof f); f[0] = 1;
    		for (int i = 0; i < m; ++ i)
    		{
    			memset(g, 0, sizeof g);
    			for (int j = 0; j < 1 << n; ++ j) if (f[j]) for (int c = 1; c <= 4; ++ c)
    				g[trans[j][c]] = (g[trans[j][c]] + f[j]) % MOD;
    			swap(f, g);
    		}
    		static int ans[N + 7];
    		memset(ans, 0, sizeof ans);
    		for (int i = 0; i < 1 << n; ++ i)
    		{
    			int cnt = 0;
    			for (int tmp = i; tmp; tmp >>= 1) if (tmp & 1) ++ cnt;
    			ans[cnt] = (ans[cnt] + f[i]) % MOD;
    		}
    		for (int i = 0; i <= n; ++ i) printf("%d
    ", ans[i]);
    	}
    }
    
  • 相关阅读:
    没有内存,怎么还能跑程序呢
    风物长宜放眼量,人间正道是沧桑
    一篇文章带你「重新认识」线程上下文切换怎么玩儿
    一文带你怼明白进程和线程通信原理
    万字长文带你还原进程和线程
    这些操作系统的概念,保你没听过!
    什么叫操作系统啊 | 战术后仰
    你要问我应用层?我就和你扯扯扯
    面试官问你MyBatis SQL是如何执行的?把这篇文章甩给他
    从这道字符串处理的难题,寻找解决复杂问题的套路
  • 原文地址:https://www.cnblogs.com/ZeonfaiHo/p/7634098.html
Copyright © 2011-2022 走看看