loj#P3284. 「USACO 2020 US Open Platinum」Exercise
「USACO 2020 US Open Platinum」Exercise
题目描述
题目译自 USACO 2020 US Open Contest, Platinum Problem 2. Exercise
Farmer John(又)想到了一个新的奶牛们的早操计划!
一如既往地,Farmer John 的 头奶牛站成一排。左数第 ()头奶牛的编号为 。FJ 让奶牛不断重复执行下列操作,直到奶牛们又回到一开始的站位为止:
- 给定一个 的排列 ,使得原位置为左数第 个的奶牛的新位置为左数第 个。
例如,如果 ,则奶牛们执行一次操作后就立刻回到了原站位。如果 ,则奶牛们需要执行 次操作才能回到原站位。每次执行完的站位分别是:
- 第 次后:。
- 第 次后:。
- 第 次后:。
- 第 次后:。
- 第 次后:。
- 第 次后:。
- 第 次后:。
请你计算对于所有 种 的排列 所需次数的乘积。
因为这个数可能非常大,所以请你输出答案对 取模的结果,保证 为质数。
下列来自 KACTL 的代码可能会对使用 C++ 语言的用户产生一定的帮助。此方法名为 Barrett reduction,它允许你使用更快的速度多次计算 ,条件是 且为常数,但无法在程序编译期被得知。
#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
}
输入格式
从标准输入中读入数据。
仅一行两个正整数 。
输出格式
输出到标准输出中。
一行一个数表示答案。
5 1000000007
369329541
数据范围与提示
对于所有数据,, 且 为质数。
对于测试点 ,满足 。
对于测试点 ,满足 。
对于测试点 ,满足 。
对于测试点 ,满足 。
对于测试点 ,无特殊限制。
出题人:Benjamin Qi