回文后缀(suffix)

回文后缀(suffix)

题目描述

给定字符集大小 SS ,问有多少个长度为 NN 的字符串不存在长度 >1>1 的回文后缀。

答案对 MM 取模。

输入格式

第一行两个正整数 n, kn,k,表示树的点数和特殊点的数量。kk 为偶数。

接下来 n − 1n1 行每行三个正整数 a, b, ca,b,c,表示 aa 和 bb 两点之间有一条边权为 cc 的无向边。

接下来一行 kk 个互不相同的 [1, n][1,n] 的正整数,为 kk 个特殊点的编号。

输入格式

一行三个数,分别为N, S, MN,S,M。

输出格式

一个数表示答案。


solution
考虑从后往前加数。
令g[i]表示后i位符合要求的串个数。
考虑从gi-1转移到gi,我们用随便放的方案减去不合法的。
而不合法的只可能是i~n为回文串
那么取f[i]=g[1+i/2],i/2即为回文串一半
g[i]=S*g[i-1]-f[i-1] 即可。
#include<cstdio>
#include<iostream>
#include<cstdlib>
#include<cstring>
#include<algorithm>
#include<cmath>
#define ll long long
#define maxn 10000007
using namespace std;
int n; 
ll S,mod,f[maxn],g[maxn];
int main()
{
    cin>>n>>S>>mod;
    f[1]=g[1]=S;
    for(int i=2;i<=n;i++){
        g[i]=(S*g[i-1])%mod-f[i-1];
        g[i]%=mod;
        f[i]=g[(i/2)+1];    
    }
    g[n]=(g[n]%mod+mod)%mod;
    cout<<g[n]<<endl;
    return 0;
}
View Code
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值