记录编号 168283 评测结果 AAAAAAAAAAAAAAAAAAAA
题目名称 [国家集训队2011]礼物(魏铭) 最终得分 100
用户昵称 GravatarRP++ 是否通过 通过
代码语言 C++ 运行时间 0.154 s
提交时间 2015-07-03 09:19:55 内存使用 0.31 MiB
显示代码纯文本
  1. #include <iostream>
  2. #include <cstdio>
  3.  
  4. using namespace std;
  5.  
  6. #define LL long long
  7.  
  8. int a[50], b[50], c[50], d[50];
  9. LL ans, tim;
  10.  
  11. int Quick_Pow(LL a, int b, int p) {
  12. LL ans = 1;
  13. while(b) {
  14. if(b&1) (ans *= a) %= p;
  15. b >>= 1, (a *= a) %= p;
  16. }
  17. return ans;
  18. }
  19.  
  20. void exgcd(int a, int b, LL &x, LL &y) {
  21. if(b == 0) {
  22. x = 1, y = 0;
  23. } else {
  24. exgcd(b, a % b, x, y);
  25. LL tmp = x;
  26. x = y, y = tmp - (a / b) * y;
  27. }
  28. }
  29.  
  30. void fact(int n, int i) {
  31. LL tmp = 1;
  32. for(int j = 2; j < d[i]; j++) if(j % a[i] != 0)
  33. (tmp *= j) %= d[i];
  34. (ans *= Quick_Pow(tmp, n / d[i], d[i])) %= d[i];
  35. for(int j = n % d[i]; j >= 2; j--) if(j % a[i] != 0)
  36. (ans *= j) %= d[i];
  37. tim += n / a[i];
  38. if(n > a[i]) fact(n / a[i], i);
  39. }
  40.  
  41. int getc(int n, int m, int i) {
  42. LL p, q, now = 0;
  43. ans = 1, tim = 0, fact(n, i);
  44. p = ans, now += tim;
  45. ans = 1, tim = 0, fact(m, i);
  46. q = ans, now -= tim;
  47. ans = 1, tim = 0, fact(n - m, i);
  48. (q *= ans) %= d[i], now -= tim;
  49. if(now >= b[i]) return 0;
  50. LL x, y;
  51. exgcd(q, d[i], x, y);
  52. if(x < 0) x += d[i];
  53. return p * Quick_Pow(a[i], now, d[i]) * x % d[i];
  54. }
  55.  
  56. LL solve(int n, int m) {
  57. for(int i = 1; i <= a[0]; i++)
  58. c[i] = getc(n, m, i);
  59. int last = d[1]; LL x, y;
  60. for(int i = 2; i <= a[0]; i++) {
  61. exgcd(last, d[i], x, y);
  62. if(y < 0) y += last;
  63. last *= d[i], (y *= c[i] - c[i - 1]) %= last, c[i] = (-y * d[i] + c[i]) % last;
  64. }
  65. return c[a[0]];
  66. }
  67.  
  68. int main() {
  69. freopen("nt2011_gift.in", "r", stdin);
  70. freopen("nt2011_gift.out", "w", stdout);
  71. int n, m, p;
  72. scanf("%d%d%d", &p, &n, &m);
  73. int tmp = p;
  74. for(int i = 2; i * i <= tmp; i++) if(tmp % i == 0) {
  75. a[++a[0]] = i;
  76. while(tmp % i == 0) {
  77. b[a[0]]++, tmp /= i;
  78. }
  79. }
  80. if(tmp > 1) a[++a[0]] = tmp, b[a[0]] = 1;
  81. for(int i = 1; i <= a[0]; i++) d[i] = Quick_Pow(a[i], b[i], p + 1);
  82. LL ans = 1;
  83. for(int i = 1, x; i <= m; i++) {
  84. scanf("%d", &x);
  85. if(n >= x) (ans *= solve(n, x)) %= p;
  86. else {
  87. printf("Impossible\n"); return 0;
  88. }
  89. n -= x;
  90. }
  91. printf("%lld\n", (ans + p) % p);
  92. }