Logo HelloWorld信息学奥赛题库

少儿编程

时间限制:1 s 空间限制:512 MB

#3983. 「USACO 2020 US Open Platinum」Exercise

Statistics

题目描述

题目译自 USACO 2020 US Open Contest, Platinum Problem 2. Exercise

Farmer John(又)想到了一个新的奶牛们的早操计划!

一如既往地,Farmer John 的 $N$ 头奶牛站成一排。左数第 $i$($1 \le i \le N$)头奶牛的编号为 $i$。FJ 让奶牛不断重复执行下列操作,直到奶牛们又回到一开始的站位为止:

  • 给定一个 $1 \sim N$ 的排列 $A$,使得原位置为左数第 $i$ 个的奶牛的新位置为左数第 $A_i$ 个。

例如,如果 $A = (1, 2, 3, 4, 5)$,则奶牛们执行一次操作后就立刻回到了原站位。如果 $A = (2, 3, 1, 5, 4)$,则奶牛们需要执行 $6$ 次操作才能回到原站位。每次执行完的站位分别是:

  • 第 $0$ 次后:$(1, 2, 3, 4, 5)$。
  • 第 $1$ 次后:$(3, 1, 2, 5, 4)$。
  • 第 $2$ 次后:$(2, 3, 1, 4, 5)$。
  • 第 $3$ 次后:$(1, 2, 3, 5, 4)$。
  • 第 $4$ 次后:$(3, 1, 2, 4, 5)$。
  • 第 $5$ 次后:$(2, 3, 1, 5, 4)$。
  • 第 $6$ 次后:$(1, 2, 3, 4, 5)$。

请你计算对于所有 $N!$ 种 $1 \sim N$ 的排列 $A$ 所需次数的乘积。

因为这个数可能非常大,所以请你输出答案对 $M$ 取模的结果,保证 $M$ 为质数。

下列来自 KACTL 的代码可能会对使用 C++ 语言的用户产生一定的帮助。此方法名为 Barrett reduction,它允许你使用更快的速度多次计算 $a \bmod b$,条件是 $b > 1$ 且为常数,但无法在程序编译期被得知。

#include <bits/stdc++.h>
using namespace std;

typedef unsigned long long ull;
typedef __uint128_t L;
struct FastMod {
    ull b, m;
    FastMod(ull b) : b(b), m(ull((L(1) << 64) / b)) {}
    ull reduce(ull a) {
        ull q = (ull)((L(m) * a) >> 64);
        ull r = a - q * b; // can be proven that 0 <= r < 2*b
        return r >= b ? r - b : r;
    }
};
FastMod F(2);

int main() {
    int M = 1000000007; F = FastMod(M);
    ull x = 10ULL*M+3; 
    cout << x << " " << F.reduce(x) << "\n"; // 10000000073 3
}

输入格式

从标准输入中读入数据。

仅一行两个正整数 $N, M$。

输出格式

输出到标准输出中。

一行一个数表示答案。

样例

input

5 1000000007

output

369329541

数组中的第 $i$ 个元素表示需要花费 $i$ 步的排列数量:$[1, 25, 20, 30, 24, 20]$。所以答案为 $1^1 \cdot 2^{25} \cdot 3^{20} \cdot 4^{30} \cdot 5^{24} \cdot 6^{20} \equiv 369329541 \pmod{{10}^9 + 7}$。

数据范围与提示

对于所有数据,$1 \le N \le 7500$,${10}^8 \le M \le {10}^9 + 7$ 且 $M$ 为质数。

对于测试点 $2$,满足 $N = 8$。
对于测试点 $3 \sim 5$,满足 $N \le 50$。
对于测试点 $6 \sim 8$,满足 $N \le 500$。
对于测试点 $9 \sim 12$,满足 $N \le 3000$。
对于测试点 $13 \sim 16$,无特殊限制。

出题人:Benjamin Qi