Gym101889J. Jumping frog(合数分解+环形dp预处理)

比赛链接:传送门

题目大意:

  一只青蛙在长度为N的字符串上跳跃,“R”可以跳上去,“P”不可以跳上去。

  字符串是环形的,N-1和0相连。

  青蛙的跳跃距离K的取值范围是[1, N-1],选定K之后不可改变。

  要求青蛙最后能跳回起点(起点可以是0-N-1的任意一个位置),问K的取值有多少种选择。

  3≤N≤105

思路:

  考虑到如果gcd(N, K) = g,则从起点开始跳的话,所有经过的点都是g的倍数,而且每个g的倍数都会经过。

  所以只要考虑从任意一个点i开始,步长为g地跳,能不遇见"P"而跳到N+i的位置的话,那么这个K可以选。

  直接模拟的话就O(N2logN)了,考虑优化。

  因为对于确定的g,对应的有很多个K,而这些K的选与不选是确定的,所以考虑枚举g(其实就是N的约数),对每个g预处理出它是否能满足题意地完成条件。

  没记错的话约数的数量应该是logN级别的,所以环形dp预处理的复杂度为O(NlogN)。

  然后枚举一遍K,更新答案就可以了。

  复杂度O(NlogN + NlogN)。

代码:

#include <bits/stdc++.h>

using namespace std;
const int MAX_N = 1e5 + 5;

int N;
char S[MAX_N];

vector <int> factor;
void getFactors(int N)
{
    factor.clear();
    for (int i = 1; i <= N/i; i++) {
        if (N%i == 0) {
            factor.push_back(i);
            if (i*i != N)
                factor.push_back(N/i);
        }
    }
    sort(factor.begin(), factor.end());
}

bool f[MAX_N << 1][50];
bool can_jump[MAX_N];
void dp()
{
    int cnt = factor.size();
    memset(f, false, sizeof f);
    for (int i = 1; i <= 2*N; i++) {
        for (int j = 0; j < cnt; j++) {
            int tmp = factor[j];
            if (S[(i-1)%N] == 'P')
                f[i][j] = false;
            else if (S[(i-1)%N] == 'R') {
                if (i-tmp <= 0)
                    f[i][j] = true;
                else
                    f[i][j] = f[i-tmp][j];
            }
        }
    }

    memset(can_jump, false, sizeof can_jump);
    for (int i = N+1; i <= 2*N; i++) {
        for (int j = 0; j < cnt; j++) {
            int tmp = factor[j];
            if (f[i][j])
                can_jump[tmp] = true;
        }
    }
}

inline int gcd(int a, int b)
{
    return a%b ? gcd(b, a%b) : b;
}

int main()
{
    scanf("%s", S);
    N = strlen(S);
    getFactors(N);
    dp();

    int ans = 0;
    for (int k = 1; k <= N-1; k++) {
        int g = gcd(k, N);
        if (can_jump[g])
            ans++;
    }
    cout << ans << endl;
    return 0;
}
View Code
原文地址:https://www.cnblogs.com/Lubixiaosi-Zhaocao/p/9980315.html