SPOJ: Large party

由polya定理$$Ans=\frac{\sum_{d|n}f(d)*\phi(\frac{n}{d})}{n}$$

$f(i)$表示不考虑旋转同构下长度为$i$的环的合法方案数。

$g(i)$表示第$i$位为男的链的方案数。

$h(i)$表示第$1$位和第$i$位都是男的链的方案数。

对于$g$数组可以$O(n)$算出来。

显然$h(i)=g(i-1)$。对于$i<=m,f(i)=2^i$,否则

$f(i)=\sum_{j=0}^m (j+1)*h(i-j)$。写出$f(i+1)$的式子,两式做差可以得到$f$数组的递推公式。

$f(i+1)=f(i)+\sum_{j=-1}^{m-1}h(i-j)-(m+1)*h(i-m)$。

当循环节$i\leq m$的时候,该循环节全都是女会导致$f(i*j)$可能不会包含$f(i)$中的这个循环节。

此时把下标大于$m$的$f$都$+1$,最后答案$-1$

 1 #include <bits/stdc++.h>
 2 using namespace std;
 3 #define MOD 100000007
 4 int phi[1010];
 5 inline int Power(int x, int y) {
 6     int ret = 1;
 7     while(y) {
 8         if(y & 1) ret = 1ll * ret * x % MOD;
 9         x = 1ll * x * x % MOD; y >>= 1;
10     }
11     return ret;
12 }
13 inline int get_phi(int x) {
14     int ret = 1;
15     for(int i = 2; i * i <= x; ++ i) {
16         if(x % i == 0) {
17             ret *= (i - 1); x /= i;
18             while(x % i == 0) {
19                 x /= i;
20                 ret *= i;
21             }
22         }
23     }
24     if(x != 1) ret *= (x - 1); 
25     return ret;
26 }
27 inline void init() {
28     for(int i = 1; i <= 1000; ++ i) {
29         phi[i] = get_phi(i);
30     }
31 }
32 int f[1010], g[1010], h[1010];
33 int sum_g[1010], sum_h[1010];
34 inline void solve() {
35     int n, m;
36     scanf("%d%d", &n, &m);
37     g[0] = 1, sum_g[0] = 1;
38     for(int i = 1; i <= n; ++ i) {
39         g[i] = sum_g[i - 1] - (i - m - 1 > 0 ? sum_g[i - m - 2] : 0);
40         if(g[i] < 0) g[i] += MOD;
41         sum_g[i] = sum_g[i - 1] + g[i];
42         if(sum_g[i] >= MOD) sum_g[i] -= MOD;
43     }
44     h[0] = 1; sum_h[0] = 1;
45     for(int i = 1; i <= n; ++ i) {
46         h[i] = g[i - 1];
47         sum_h[i] = sum_h[i - 1] + h[i];
48         if(sum_h[i] >= MOD) sum_h[i] -= MOD;
49     }
50     f[0] = 1;
51     for(int i = 1; i <= n; ++ i) {
52         if(i <= m) f[i] = f[i - 1] * 2 % MOD;
53         else if(i == m + 1) {
54             f[i] = 2 * f[i - 1] - 1;
55             f[i] %= MOD;
56         }
57         if(i >= m + 1) {
58             f[i + 1] = f[i] + sum_h[i + 1] - sum_h[i - m] - 1ll * (m + 1) * h[i - m] % MOD + MOD + MOD;
59             f[i + 1] %= MOD;
60         }
61     }
62     for(int i = m + 1; i <= n; ++ i) {
63         f[i] ++;
64         if(f[i] >= MOD) f[i] -= MOD;
65     } 
66     int ans = 0;
67     for(int i = 1; i * i <= n; ++ i) if(n % i == 0) {
68         ans += 1ll * f[i] * phi[n / i] % MOD;
69         if(ans >= MOD) ans -= MOD;
70         if(i != n / i) {
71             ans += 1ll * f[n / i] * phi[i] % MOD;
72             if(ans >= MOD) ans -= MOD;
73         }
74     }
75     printf("%d\n", 1ll * ans * Power(n, MOD - 2) % MOD - (m >= n ? 0 : 1));
76 }
77 int main() {
78     init();
79     int T;
80     scanf("%d", &T);
81     while(T --) {
82         solve();
83     }
84 }
View Code
posted @ 2019-12-14 20:19  iamunstoppable  阅读(428)  评论(0编辑  收藏  举报