• 【洛谷P4838】P哥破解密码【矩阵乘法】


    题目大意:

    题目链接:https://www.luogu.org/problemnew/show/P4838
    求长度为nn的只包含ABAB的字符串且没有任意三个AA相连的字符串个数。


    思路:

    因为有不能任意三个AA相连的限制,所以可以讨论长度为nn的字符串的后缀方案及对n+1n+1的影响。
    首先,长度为nn的字符串后缀总数只有3个:

    • 后缀为baba。这样显然只能由倒数第二个位置为bb的字符串得来。
    • 后缀为aaaa。这样显然只能由倒数第二个位置为aa的字符串得来。
    • 后缀为bb。这样显然去掉bb后的后缀为ba,aaba,aa还是bb都是无所谓的。

    f[n][s]f[n][s]表示长度为nn的后缀为ss的字符串的个数,那么有
    f[n][ba]=f[n1][b]f[n][ba]=f[n-1][b]
    f[n][aa]=f[n1][a]f[n][aa]=f[n-1][a]
    f[n][b]=f[n1][ba]+f[n1][aa]+f[n1][b]f[n][b]=f[n-1][ba]+f[n-1][aa]+f[n-1][b]
    所以我们可以构造出如下矩阵:
    在这里插入图片描述
    然后每次询问都跑一边矩阵乘法就行了。


    代码:

    #include <cstdio>
    #include <cstring>
    using namespace std;
    typedef long long ll;
    
    const ll MOD=19260817;
    int n,m;
    
    struct matrix
    {
    	ll a[4][4];
    }f,A,a;
     
    matrix operator *(matrix a,matrix b)
    {
    	matrix c;
    	memset(c.a,0,sizeof(c.a));
    	for(int i=1;i<=3;i++)
    		for(int j=1;j<=3;j++)
    	  		for(int k=1;k<=3;k++)
    	      		c.a[i][j]=(c.a[i][j]+a.a[i][k]*b.a[k][j])%MOD;
    	return c;
    }
    
    void power(int M)
    {
    	for (;M;M>>=1,a=a*a)
    		if (M&1) f=f*a;
    }
    
    int main()
    {
    	A.a[3][1]=A.a[1][2]=A.a[1][3]=A.a[2][3]=A.a[3][3]=1;
    	scanf("%d",&m);
    	while (m--)
    	{
    		memcpy(a.a,A.a,sizeof(A.a));
    		memset(f.a,0,sizeof(f.a));
    		f.a[1][1]=f.a[1][3]=1;
    		scanf("%d",&n);
    		power(n-1);
    		printf("%lld
    ",(f.a[1][1]+f.a[1][2]+f.a[1][3])%MOD);
    	}
    	return 0;
    }
    
  • 相关阅读:
    Oracle 实例恢复
    使用 ASMCMD 工具管理ASM目录及文件
    软考编译原理总结
    解决初学者学不懂android,不理解android的设计
    对计算机模拟人脑的一个小想法
    [每日一题] 11gOCP 1z0-052 :2013-09-19 创建用户...................................................B41
    andengine游戏引擎总结基础篇
    hdu 1789 Doing Homework again (贪心)
    poj 3026 (最小生成树)
    autolisp 列表 resbuf
  • 原文地址:https://www.cnblogs.com/hello-tomorrow/p/11998111.html
Copyright © 2020-2023  润新知