Z城市居住着很多只跳蚤。在Z城市周六生活频道有一个娱乐节目。一只跳蚤将被请上一个高空钢丝的正中央。钢丝很长,可以看作是无限长。节目主持人会给该跳蚤发一张卡片。卡片上写有N+1个自然数。其中最后一个是M,而前N个数都不超过M,卡片上允许有相同的数字。跳蚤每次可以从卡片上任意选择一个自然数S,然后向左,或向右跳S个单位长度。而他最终的任务是跳到距离他左边一个单位长度的地方,并捡起位于那里的礼物。比如当N=2,M=18时,持有卡片(10, 15, 18)的跳蚤,就可以完成任务:他可以先向左跳10个单位长度,然后再连向左跳3次,每次15个单位长度,最后再向右连跳3次,每次18个单位长度。而持有卡片(12, 15, 18)的跳蚤,则怎么也不可能跳到距他左边一个单位长度的地方。当确定N和M后,显然一共有MN张不同的卡片。现在的问题是,在这所有的卡片中,有多少张可以完成任务。
输入文件有且仅有一行,包括用空格分开的两个整数N和M。
题解
该死的bz(婊子)oj没用官方数据,还要用高精度。
一张可行的卡片,就是将上面的数字经过加减变换能够得到$1$,于是我们可以猜想一下卡片上的数字所需满足的要求。首先很容易猜到,这些数字必然有奇数有偶数,但是仅满足这个条件是显然不够的,如$3$,$6$就无法得出$1$来,于是可以继续猜想这些数字两两互质,即所有数字的公因数为$1$,用数论的方法可以证明这个猜想是正确的。然后可以用容斥原理来算出答案,这里举个例子来说明,假设$m=30=2*3*5$,答案=$m^n$-(有公因数$2$的$n$元组)-(有公因数$3$的$n$元组)-(有公因数$5$的$n$元组)+(有公因数$2$,$3$的$n$元组)+(有公因数$2$,$5$的$n$元组)+(有公因数$3$,$5$的$n$元组)-(有公因数$2$,$3$,$5$的$n$元组)
1 #include <set>
2 #include <map>
3 #include <ctime>
4 #include <cmath>
5 #include <queue>
6 #include <stack>
7 #include <vector>
8 #include <cstdio>
9 #include <string>
10 #include <cstring>
11 #include <cstdlib>
12 #include <iostream>
13 #include <algorithm>
14 #define LL long long
15 #define Max(a, b) ((a) > (b) ? (a) : (b))
16 #define Min(a, b) ((a) < (b) ? (a) : (b))
17 using namespace std;
18 const int INF = ~0u>>1;
19
20 LL n, m;
21 LL ans = 0;
22 LL q[1005], tot;
23
24 LL pow(LL a, LL b){
25 LL c = 1;
26 while (b){
27 if (b&1) c *= a;
28 b >>= 1;
29 a *= a;
30 }
31 return c;
32 }
33 void sperate(LL m){
34 for (LL i = 2; i*i <= m; i++)
35 if (m%i == 0){
36 q[++tot] = i;
37 while (m%i == 0) m /= i;
38 }
39 if (m-1) q[++tot]=m;
40 }
41 void dfs(int cen, LL cnt, int cho){
42 if (cen > tot){
43 if (cho == 0) return;
44 if (cho%2) ans -= pow(m/cnt, n);
45 else ans += pow(m/cnt, n);
46 return;
47 }
48 dfs(cen+1, cnt, cho);
49 dfs(cen+1, cnt*q[cen], cho+1);
50 }
51
52 int main(){
53 scanf("%lld%lld", &n, &m);
54 ans = pow(m, n);
55 sperate(m);
56 dfs(1, 1, 0);
57 printf("%lld\n", ans);
58 return 0;
59 }