天天看点

[hdu 5351] MZL's Border

题目链接,我怎么知道题目的标题是什么意思?

题目大意:

定义字符串 s1=a , s2=b , si=si−1+si−2  (i≥3)

同时定义一个字符串 s 的权值为一个最大的 i<|s| 满足 s 长度为 i 的前缀等于长度为 i 的后缀。

比如字符串 aba 的权值就是 1 ,abab 的权值就是 2 ,aaaa 的权值就是 3 。

求 sn 的前 m 个字符组成的字符串的权值,输出权值取模 258280327 的结果。

首先很显然的一点是, si 包含 si−1  (i≥3)

那么我们考虑 n≥3 的情况, 设 fi 表示 sn 的前 i 个字符组成的字符串的权值

然后用 KMP 求出 next 数组得到 {fi}  (n=19) :

{fi}={0,0,1,1,2,3,2,3,4,5,6,4,5,6,7,8,9,10,11}

将连续的数字分成一段:

0   0 1   1 2 3   2 3 4 5 6   4 5 6 7 8 9 10 11

分成的段的长度分别为 1,2,3,5,8 ,起始值分别为 0,0,1,2,4

可以发现,每一段长度都是 fibi+1 ,起始值都是 fibi−1 ( fibi 为斐波拉契数列 )

大胆猜想,小心假设, 从不证明!

与暴力对拍之后发现这个规律是对的。

考虑 next[] 数组的构造即可证明这个结论。

根据这个规律可以 O(n) 求出答案,然后就做完了

#include <cstdio>
#include <cstdlib>
#include <cstring>
#include <cmath>
#include <ctime>
#include <vector>
#include <utility>
#include <stack>
#include <queue>
#include <map>
#include <iostream>
#include <algorithm>

template<class Num>void read(Num &x)
{
    char c; int flag = ;
    while((c = getchar()) < '0' || c > '9')
        if(c == '-') flag *= -;
    x = c - '0';
    while((c = getchar()) >= '0' && c <= '9')
        x = (x<<) + (x<<) + (c-'0');
    x *= flag;
    return;
}
template<class Num>void write(Num x)
{
    if(!x) {putchar('0');return;}
    if(x < ) putchar('-'), x = -x;
    static char s[];int sl = ;
    while(x) s[sl++] = x% + '0',x /= ;
    while(sl) putchar(s[--sl]);
}
const int maxn = , maxL = , Base = , Mod = ;
int n;

struct BigNum
{
    int len, num[maxL]; 

    void read()
    {
        static char s[maxL];
        scanf("%s", s + );
        len = strlen(s + );
        for(int i = len; i >= ; i--)
            num[i] = s[len +  - i] - '0';
    }
    void write()
    {
        if(!len)
        {
            putchar('0');
            return;
        }
        for(int i = len; i >= ; i--)
            putchar(num[i] + '0');
    }

}m, F[maxn + ], empty, one, two;

BigNum operator + (const BigNum &a,const BigNum &b)
{
    BigNum c = empty;
    c.len = std::max(a.len, b.len);
    for(int i = ; i <= c.len; i++)
        c.num[i] = a.num[i] + b.num[i];
    for(int i = ; i <= c.len; i++)
    {
        c.num[i + ] += c.num[i] / Base;
        c.num[i] %= Base;
    }
    while(c.num[c.len + ])
    {
        ++c.len;
        c.num[c.len + ] += c.num[c.len] / Base;
        c.num[c.len] %= Base;
    }
    return c;
}
BigNum operator - (const BigNum &a,const BigNum &b)
{
    BigNum c = empty;
    for(int i = ; i <= a.len; i++)
        c.num[i] = a.num[i] - b.num[i];
    for(int i = ; i <= a.len; i++)
    {
        if(c.num[i] < )
        {
            c.num[i + ] --;
            c.num[i] += Base;
        }
    }
    c.len = ;
    for(int i = a.len; i >= ; i --)
    {
        if(c.num[i])
        {
            c.len = i;
            break;
        }
    }
    return c;
}
bool operator > (const BigNum &a,const BigNum &b)
{
    if(a.len != b.len) return a.len > b.len;
    for(int i = a.len; i >= ; i--)
    {
        if(a.num[i] != b.num[i])
            return a.num[i] > b.num[i];
    }
    return false;
}
int operator %(const BigNum &a,int mod)
{
    int r = ;
    for(int i = a.len; i >= ; i--)
        r = ((long long)r * Base + a.num[i])%Mod;
    return r;
}

void prework()
{
    one.num[] = one.len = ;
    two.num[] = , two.len = ;

    F[] = F[] = one;
    for(int i = ; i <= maxn; i++)
        F[i] = F[i - ] + F[i - ];
}
void init()
{
    read(n), m.read();
}
void solve()
{
    for(int i = ; i <= n; i++)
    {
        if(m > F[i])
            m = m - F[i];
        else
        {
            write((F[i - ] + m - two)%Mod);
            puts("");
            return;
        }
    }
}

int main()
{
    int T;

    read(T);
    prework();

    while(T--) init(), solve();

    return ;
}
           
HDU