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

本文详细解析洛谷P4838题目,该题要求计算长度为n的仅包含A和B的字符串数量,且无任意三个连续A出现。通过分析字符串后缀及其对后续字符的影响,采用矩阵乘法高效求解。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

题目大意:

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


思路:

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

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

f [ n ] [ s ] f[n][s] f[n][s]表示长度为 n n n的后缀为 s s s的字符串的个数,那么有
f [ n ] [ b a ] = f [ n − 1 ] [ b ] f[n][ba]=f[n-1][b] f[n][ba]=f[n1][b]
f [ n ] [ a a ] = f [ n − 1 ] [ a ] f[n][aa]=f[n-1][a] f[n][aa]=f[n1][a]
f [ n ] [ b ] = f [ n − 1 ] [ b a ] + f [ n − 1 ] [ a a ] + f [ n − 1 ] [ b ] f[n][b]=f[n-1][ba]+f[n-1][aa]+f[n-1][b] f[n][b]=f[n1][ba]+f[n1][aa]+f[n1][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\n",(f.a[1][1]+f.a[1][2]+f.a[1][3])%MOD);
	}
	return 0;
}
洛谷P1928 外星密码是一道经典的字符串处理目大致描述是这样的:给定一段由外星人发出的信号序列(即一串字母组成的字符串),需要从中找出最长的一个连续子串满足某些特定条件——比如每个字母出现次数均为偶数次或者其他指定规则下的统计特性。 以下是用C语言解决这个问的基本思路及步骤: ### 解决方案概述 #### 第一步:输入数据 首先读入代表“外星密码”的整个字符串。通常我们会限制这个字符串长度不超过一定值(如目设定的最大限制)。例如可以用下面的方法来进行存储: ```c char str[MAX_LEN]; // MAX_LEN 应设置得足够大以容纳最大可能的输入大小+1 (用于&#39;\0&#39;) scanf("%s", str); // 直接利用scanf读取完整的单词型字符串 ``` #### 第二步:遍历寻找符合条件子串 我们采用双层循环的方式依次检查每一个可能的起始位置i以及结束位置j所形成的子串是否符合要求。对于每次选定的[i,j]区间内的部分,都需要计数其中各个字符的数量,并验证这些数字是否全部为偶数或其他预设的标准。 ```c for(int i = 0; i < strlen(str); ++i){ int count[ALPHA_SIZE]={0}; // ALPHA_SIZE一般设成26适应小写字母表情况 for(int j=i ; j<strlen(str) && valid<=bestValid; j++){ updateCount(count,str[j]); if(checkCondition(count)){ recordBestResult(i,j,...); } } } ``` 这里的伪代码展示了两重嵌套迭代框架结构,内侧随着索引推进不断调整局部频率直方图(`count`)同时测试目标属性达成状况;一旦发现更优结果则保存下来直到最终确定全局最佳解法为止。 #### 实现细节补充说明 - `updateCount`: 增量更新当前考虑范围内某个字符对应的频度计量项数值; - `checkCondition`: 根据实际问需求定义判断依据函数检验目前收集到的信息集合能否构成合格候选答案; - `recordBestResult`: 当前找到新的较理想匹配区域时记录相关信息供后续比较参考之需。 完整版解决方案可以根据具体约束进一步优化时间复杂度或空间利用率等方面的表现特征。 ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值