hdu4944FSF’s game(数论)

来源:互联网 发布:lol检测到账号数据异常 编辑:程序博客网 时间:2024/05/17 23:02

题目:hdu4944FSF’s game(数论)


题目大意:给定N,然后会有N * ( N + 1)/2个等级的矩形,(1 * 1, 2 * 1, 2* 2, ...N * 1, N * 2.. N* N).将这些矩阵分成大小相同的K * K 的正方形,能够获得金币A * B / gcd(A
/ K, B/ K);然后给定N,问能够得到的总金币。


解题思路:

                 对于sum(N):N * (N + 1) / 2个等级的矩形的金币总和 = sum(N - 1) + f(N):(N * 1, N * 2,...N * N)这一些矩形的金币之和)。

                 例如N = 8: 

                                      1* 8/ gcd(8/1, 1/1)  2* 8 / gcd(8/1,2/1) 2 * 8/ gcd(8/2,  2/2) ....8 * 8 / gcd(8/1,8/2) 8 * 8/ gcd(8/2.8/2), 8 * 8/ gcd(8/4, 8/4), 8 * 8 / gcd(8/8, 8/8)

                                      [1* 8    2* 8/2 @    2*8     3* 8     4 * 8    4 * 8/2@    4 * 8/4     5 * 8    6 * 8    6*8 /2 @   7 * 8    8 * 8   @8 * 8/2   8 *8/4   8*8 /8]       

                                      可以发现gcd(A /K, B/K)是N的因子。另a = gcd, N = n*a.

                                      对于gcd = a,可获得的金币G(a):1*a * N/ a   + 2 * a * N / a + 3 *a * N/a +...+ na * N /a = (1 +2 + 3 + n) * N

                                      那么枚举N的因子n,加起来就是f(N) = sum1..n(G(ai)).


代码:

#include <cstdio>#include <cstring>typedef unsigned __int64 ll;const int maxn = 500005;const ll M = 1LL<<32;//int1会溢出ll f1[maxn];ll f2[maxn];void init () {for (int i = 1; i < maxn; i++) {for (int j = 1; j * i < maxn; j++) f1[i * j] = (f1[i * j] + (1LL + i) * i / 2) % M;}f2[1] = 1;for (int i = 2; i < maxn; i++)f2[i] = (f2[i - 1] + f1[i] * i) % M; }int main () {int t;int n;init ();scanf ("%d", &t);for (int i = 1; i <= t; i++) {scanf ("%d", &n);printf ("Case #%d: %I64u\n", i, f2[n]); }return 0;}


0 0
原创粉丝点击