forked from ckcz123/codejam
-
Notifications
You must be signed in to change notification settings - Fork 0
/
D.java
78 lines (72 loc) · 2.34 KB
/
D.java
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
import java.io.PrintStream;
import java.util.*;
/**
* APAC 2017 Round B Problem D: Sherlock and Permutation Sorting
* Check README.md for explanation.
*/
public class Main {
private long mode;
private long[] prims;
private long[] facts;
/**
* 简要算法:
* 设prims[n]为长度为n的不能再分割的排列个数(前几项为 0, 1, 1, 3, 13, ...)
* 则prims[n]=n! - sigma_(i,0,n-1) {i!*prims[n-i]}
* 设f[n, k]代表长度为n的且最多被分割为k块的排列个数
* 则f[n, 1]=prims[n]
* f[n, k] = sigma_(i,0,n-1) {f[i, k-1]*prims[n-i]}
* 我们要求的是 ss[n] = sigma_(k,1,n) {k*k*f[n, k]}
* 首先令s[n] = sigma_(k,1,n) {k*f[n, k]}
* 则s[n] = sigma_(i,0,n-1) {(s[i]+i!)*prims[n-i]}
* 进而ss[n] = sigma_(i,0,n-1) {(ss[i]+2*s[i]+i!)*prims[n-i]}
* 即为所求,O(n^2)
*/
public String solve(Scanner scanner) {
int n=scanner.nextInt();
mode=scanner.nextLong();
calFacts(n);
calPrim(n);
long[] s=new long[n+1], ss=new long[n+1];
for (int i=1;i<=n;i++) {
for (int j=0;j<=i-1;j++) {
s[i]+=prims[i-j]*(facts[j]+s[j]);
s[i]%=mode;
}
}
for (int i=1;i<=n;i++) {
for (int j=0;j<=i-1;j++) {
ss[i]+=prims[i-j]*(ss[j]+2*s[j]+facts[j]);
ss[i]%=mode;
}
}
return String.valueOf(ss[n]);
}
private void calFacts(int n) {
facts=new long[n+1];
facts[0]=1;
for (int i=1;i<=n;i++)
facts[i]=facts[i-1]*i%mode;
}
private void calPrim(int n) {
prims=new long[n+1];
for (int i=1;i<=n;i++) {
long u=0;
for (int j=1;j<=i-1;j++) {
u+=facts[j]*prims[i-j];
u%=mode;
}
prims[i]=(facts[i]-u+mode)%mode;
}
}
public static void main(String[] args) throws Exception {
System.setOut(new PrintStream("output.txt"));
Scanner scanner=new Scanner(System.in);
int times=scanner.nextInt();
long start=System.currentTimeMillis();
for (int t=1;t<=times;t++) {
System.out.println(String.format("Case #%d: %s", t, new Main().solve(scanner)));
}
long end=System.currentTimeMillis();
System.err.println(String.format("Time used: %.3fs", (end-start)/1000.));
}
}