【ARC064-F】【XSY2575】Rotated Palindromes(DP)(字符串)

Description
然而,由于小C沉迷于制作游戏,他完全忘记了自己作为国家集训队的一员,还有156道作业题等他完成。还有一天作业就要截止了,而他一题还没有做。于是他赶紧挑了一道看起来最简单的题:

“给定一个整数N,请你求出有多少字符集为1到K之间整数的字符串,使得该字符串可以由一个长度为N的回文串循环移位后得到。所谓循环移位,就是把字符串的某个前缀(可以为空)移到字符串末尾,如"1221"循环移位可以得到"1221"、“2211”、“2112”、"1122"四个字符串。结果对109+7取模。”

为了不让小C的集训队资格被CCF取消,请你帮助他完成这道题吧。

Input

第一行包含两个整数N,K。

Output

输出满足条件的字符串数对109+7取模的结果。

Sample Input

Sample Input 1

4 2

Sample Input 2

1 10

Sample Input 3

6 3

Sample Input 4

1000000000 1000000000

Sample Output

Sample Output 1

6

Sample Output 2

10

Sample Output 3

75

Sample Output 4

875699961

HINT

在第一个样例中,有"1111"、“1122”、“1221”、“2211”、“2112”、“2222”,共6个字符串符合条件。

我们先考虑枚举回文串,因为回文串是回文的,所以我们只用枚举回文串的一半,回文串的个数就是 m ( n + 1 ) / 2 m^{(n+1)/2} m(n+1)/2

每一个回文串应该有 n n n种循环移位后的到的字符串,但是这样计算后我们会发现这比答案大很多,因为有重复。

我们考虑如何计算不重复的字符串的个数,我们可以枚举回文串的循环节。

分两种情况讨论。

1.循环节为奇数

因为一个回文串的循环节必定为回文串,所以我们循环移位1个循环节的长度为 l e n len len就会有重复,所以会产生 l e n len len个新字符串。

2.循环节为偶数

因为上述的原因,循环节必定回文,我们只要有 l e n / 2 len/2 len/2的长度就会重复。

比如说,串 122111221 122111221 122111221,我们移了两位之后,串变成 21122112 21122112 21122112,虽然看起来与原串并不相同,但是我们在枚举回文串的时候,也会枚举到 21122112 21122112 21122112,所以也算重复。

接下来DP。

我们先预处理出 n n n的所有因数, a [ ] a[] a[]

我们设 d p [ i ] dp[i] dp[i]表示最小循环节为 a [ i ] a[i] a[i]时,有多少个符合条件的回文串。

我们在 d p [ i ] dp[i] dp[i]中减去 d p [ a [ i ] 的 因 子 ] dp[a[i]的因子] dp[a[i]],为了保证 a [ i ] a[i] a[i]时最小的循环节。

ans就直接加上回文串所贡献的字符串个数即可。

时间复杂度大约 O ( 因 子 个 数 2 ∗ l o g ( n / 2 ) ) O(因子个数^2*log(n/2)) O2log(n/2)


#include<bits/stdc++.h>
#define mod 1000000007
#define int long long
using namespace std;
int dp[2010],cnt,n,k,a[2010],ans;
int fastpow(int x,int y)
{
    int sum=1;
    while(y)
    {
        if(y&1)
        {
            sum=(1ll*sum*x)%mod;
        }
        x=(1ll*x*x)%mod;
        y>>=1;
    }
    return sum;
}
signed main()
{
    scanf("%lld%lld",&n,&k);
    for(int i=1;i*i<=n;i++)//枚举因子
    {
        if(n%i==0)
        {
            a[++cnt]=i;
            if(i*i!=n)
            {
                a[++cnt]=n/i;
            }
        }
    }
    sort(a+1,a+cnt+1);
    for(int i=1;i<=cnt;i++)
    {
        dp[i]=fastpow(k,(a[i]+1)/2);//有多少个回文串
        for(int j=1;j<i;j++)
        {
            if(a[i]%a[j]==0)//去重
            {
                dp[i]=(dp[i]-dp[j]+mod)%mod;
            }
        }
        if(a[i]&1)//统计答案
        {
            ans=(ans+dp[i]*a[i])%mod;
        }else{
            ans=(ans+dp[i]*(a[i]/2))%mod;
        }
    }
    printf("%lld\n",ans);
    return 0;
}
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值