2020-06-06 01:53:38 +08:00
|
|
|
/**
|
|
|
|
* \file
|
|
|
|
* \brief [Problem 21](https://projecteuler.net/problem=21) solution
|
2020-06-07 02:51:49 +08:00
|
|
|
* \author [Krishna Vedala](https://github.com/kvedala)
|
2020-06-06 01:53:38 +08:00
|
|
|
*/
|
2020-04-02 09:09:42 +08:00
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <time.h>
|
|
|
|
|
|
|
|
/**
|
|
|
|
* function to return the sum of proper divisors of N
|
2020-06-29 03:18:52 +08:00
|
|
|
*/
|
2020-06-06 01:53:38 +08:00
|
|
|
unsigned long sum_of_divisors(unsigned int N)
|
2020-04-02 09:09:42 +08:00
|
|
|
{
|
2020-06-06 01:53:38 +08:00
|
|
|
unsigned long sum = 1 + N; /* 1 and itself are always a divisor */
|
2020-04-02 09:09:42 +08:00
|
|
|
/* divisors are symmertically distributed about the square-root */
|
2020-06-06 01:53:38 +08:00
|
|
|
for (unsigned int i = 2; i * i < N; i++)
|
2020-04-02 09:09:42 +08:00
|
|
|
{
|
2020-06-06 01:53:38 +08:00
|
|
|
if ((N % i) != 0)
|
|
|
|
/* i is not a divisor of N */
|
2020-04-02 09:09:42 +08:00
|
|
|
continue;
|
|
|
|
|
|
|
|
// #ifdef DEBUG
|
|
|
|
// printf("%4d, %4d,", i, N / i);
|
|
|
|
// #endif
|
|
|
|
|
2020-06-06 01:53:38 +08:00
|
|
|
sum += i + (N / i);
|
2020-04-02 09:09:42 +08:00
|
|
|
}
|
|
|
|
// #ifdef DEBUG
|
|
|
|
// printf("\nSum of divisors of %4d: %4d\n", N, sum);
|
|
|
|
// #endif
|
|
|
|
return sum;
|
|
|
|
}
|
|
|
|
|
2020-06-06 01:53:38 +08:00
|
|
|
/** Main function */
|
2020-04-02 09:09:42 +08:00
|
|
|
int main(int argc, char **argv)
|
|
|
|
{
|
|
|
|
unsigned long sum = 0;
|
|
|
|
unsigned int MAX_N = 500;
|
|
|
|
if (argc == 2)
|
|
|
|
MAX_N = atoi(argv[1]);
|
|
|
|
|
2020-06-29 03:18:52 +08:00
|
|
|
/*
|
2020-04-02 09:09:42 +08:00
|
|
|
* We use an array of flags to check if a number at the index was:
|
|
|
|
* not-processed = 0
|
|
|
|
* is amicable = 1
|
|
|
|
* not amicable = -1
|
2020-06-29 03:18:52 +08:00
|
|
|
*/
|
2020-04-02 09:09:42 +08:00
|
|
|
char *flags = (char *)calloc(MAX_N, sizeof(char));
|
|
|
|
|
|
|
|
clock_t start_time = clock();
|
2020-04-24 08:59:02 +08:00
|
|
|
int i;
|
2020-04-02 09:09:42 +08:00
|
|
|
/* there are no such numbers till 10. Lets search from there on */
|
2020-04-24 08:17:32 +08:00
|
|
|
for (i = 10; i < MAX_N; i++)
|
2020-04-02 09:09:42 +08:00
|
|
|
{
|
|
|
|
if (flags[i] != 0)
|
|
|
|
/* already processed, skip */
|
|
|
|
continue;
|
|
|
|
|
|
|
|
unsigned int b = sum_of_divisors(i);
|
|
|
|
if (b >= MAX_N)
|
|
|
|
flags[i] = -1;
|
|
|
|
else if (flags[b] == -1)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
unsigned int c = sum_of_divisors(b);
|
|
|
|
if (c == i && b != i)
|
|
|
|
{
|
|
|
|
/* found amicable */
|
|
|
|
flags[b] = 1;
|
|
|
|
flags[i] = 1;
|
|
|
|
sum += b + i;
|
|
|
|
#ifdef DEBUG
|
|
|
|
printf("Amicable: %4d : %4d\n", i, b);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
flags[i] = -1;
|
|
|
|
if (b < MAX_N)
|
|
|
|
flags[b] = -1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
clock_t end_time = clock();
|
|
|
|
|
2020-05-30 04:23:24 +08:00
|
|
|
printf("\nTime taken: %.4g millisecond\n",
|
|
|
|
1e3 * (end_time - start_time) / CLOCKS_PER_SEC);
|
2020-04-02 09:09:42 +08:00
|
|
|
printf("Sum of all numbers = %lu\n", sum);
|
|
|
|
|
|
|
|
free(flags);
|
|
|
|
return 0;
|
|
|
|
}
|